2017-02-07 22:33:23 +01:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/docker/go-connections/nat"
|
|
|
|
"github.com/xeipuuv/gojsonschema"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2017-07-06 16:28:13 +02:00
|
|
|
schemaLoaderV1 gojsonschema.JSONLoader
|
|
|
|
constraintSchemaLoaderV1 gojsonschema.JSONLoader
|
|
|
|
schemaLoaderV2 gojsonschema.JSONLoader
|
|
|
|
constraintSchemaLoaderV2 gojsonschema.JSONLoader
|
|
|
|
schemaV1 map[string]interface{}
|
|
|
|
schemaV2 map[string]interface{}
|
2017-02-07 22:33:23 +01:00
|
|
|
)
|
|
|
|
|
2017-07-06 16:28:13 +02:00
|
|
|
func init() {
|
|
|
|
if err := setupSchemaLoaders(schemaDataV1, &schemaV1, &schemaLoaderV1, &constraintSchemaLoaderV1); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := setupSchemaLoaders(servicesSchemaDataV2, &schemaV2, &schemaLoaderV2, &constraintSchemaLoaderV2); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-07 22:33:23 +01:00
|
|
|
type (
|
|
|
|
environmentFormatChecker struct{}
|
|
|
|
portsFormatChecker struct{}
|
|
|
|
)
|
|
|
|
|
|
|
|
func (checker environmentFormatChecker) IsFormat(input string) bool {
|
|
|
|
// If the value is a boolean, a warning should be given
|
|
|
|
// However, we can't determine type since gojsonschema converts the value to a string
|
|
|
|
// Adding a function with an interface{} parameter to gojsonschema is probably the best way to handle this
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (checker portsFormatChecker) IsFormat(input string) bool {
|
|
|
|
_, _, err := nat.ParsePortSpecs([]string{input})
|
|
|
|
return err == nil
|
|
|
|
}
|
|
|
|
|
2017-07-06 16:28:13 +02:00
|
|
|
func setupSchemaLoaders(schemaData string, schema *map[string]interface{}, schemaLoader, constraintSchemaLoader *gojsonschema.JSONLoader) error {
|
|
|
|
if *schema != nil {
|
2017-02-07 22:33:23 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var schemaRaw interface{}
|
2017-07-06 16:28:13 +02:00
|
|
|
err := json.Unmarshal([]byte(schemaData), &schemaRaw)
|
2017-02-07 22:33:23 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2017-07-06 16:28:13 +02:00
|
|
|
*schema = schemaRaw.(map[string]interface{})
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
gojsonschema.FormatCheckers.Add("environment", environmentFormatChecker{})
|
|
|
|
gojsonschema.FormatCheckers.Add("ports", portsFormatChecker{})
|
|
|
|
gojsonschema.FormatCheckers.Add("expose", portsFormatChecker{})
|
2017-07-06 16:28:13 +02:00
|
|
|
*schemaLoader = gojsonschema.NewGoLoader(schemaRaw)
|
2017-02-07 22:33:23 +01:00
|
|
|
|
2017-07-06 16:28:13 +02:00
|
|
|
definitions := (*schema)["definitions"].(map[string]interface{})
|
2017-02-07 22:33:23 +01:00
|
|
|
constraints := definitions["constraints"].(map[string]interface{})
|
|
|
|
service := constraints["service"].(map[string]interface{})
|
2017-07-06 16:28:13 +02:00
|
|
|
*constraintSchemaLoader = gojsonschema.NewGoLoader(service)
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// gojsonschema doesn't provide a list of valid types for a property
|
|
|
|
// This parses the schema manually to find all valid types
|
|
|
|
func parseValidTypesFromSchema(schema map[string]interface{}, context string) []string {
|
|
|
|
contextSplit := strings.Split(context, ".")
|
|
|
|
key := contextSplit[len(contextSplit)-1]
|
|
|
|
|
|
|
|
definitions := schema["definitions"].(map[string]interface{})
|
|
|
|
service := definitions["service"].(map[string]interface{})
|
|
|
|
properties := service["properties"].(map[string]interface{})
|
|
|
|
property := properties[key].(map[string]interface{})
|
|
|
|
|
|
|
|
var validTypes []string
|
|
|
|
|
|
|
|
if val, ok := property["oneOf"]; ok {
|
|
|
|
validConditions := val.([]interface{})
|
|
|
|
|
|
|
|
for _, validCondition := range validConditions {
|
|
|
|
condition := validCondition.(map[string]interface{})
|
|
|
|
validTypes = append(validTypes, condition["type"].(string))
|
|
|
|
}
|
|
|
|
} else if val, ok := property["$ref"]; ok {
|
|
|
|
reference := val.(string)
|
|
|
|
if reference == "#/definitions/string_or_list" {
|
|
|
|
return []string{"string", "array"}
|
|
|
|
} else if reference == "#/definitions/list_of_strings" {
|
|
|
|
return []string{"array"}
|
|
|
|
} else if reference == "#/definitions/list_or_dict" {
|
|
|
|
return []string{"array", "object"}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return validTypes
|
|
|
|
}
|