Sails JS: model validation with built-in JSON property

How to test model with property of type object in Sails JS?

I know that attributes with a simple value (like a string) are acceptable, however how does this work for nested JSON?

As below:

{
    name: 'John',
    location: {
        x: 23,
        y: 15,
        z: 50
    }
}

So will it have the form:

{
    name: {
        type: 'string',
        required: true,
    },
    location: {
        x: {
            type: 'number',
            required: true
        },
        y: {
            type: 'number',
            required: true
        },
        z: {
            type: 'number',
            required: true
        }
    }
}
+5
source share
3 answers

Waterline (Sails ORM) does not directly support nested schemas. You can use a custom validation rule to validate an attribute:

module.exports = {

  types: {
    location: function(val) {
      // Make sure that x, y and z are present and are numbers.
      // This won't allow numeric strings, but you can adjust to fit your needs.
      return (_.isNumber(val.x) && _.isNumber(val.y) && _.isNumber(val.z));
    }
  },

  attributes: {

    location: {
      type: 'json',
      required: true, // If you want the whole attribute to be required
      location: true  // Validate that the attribute has the schema you want
    }

    ...more attributes...

  }

};
+5
source

, , "machine" ( sailsjs, sailsjs) "" .

:

$ cat api/types/define.js 
const buildWithCustomUsage = require("machine").buildWithCustomUsage;

function validateWith(machine, inputs) {
    machine(inputs).now();
    return true;
}

module.exports = function (def) {
    const machine = buildWithCustomUsage({
        def,
        extraArginsTactic: "doNotCheck"
    });
    return {
        machine,
        validate: validateWith.bind(null, machine)
    };
};

:

$ cat api/types/QuoteRequest.js 
module.exports = require("./define")({

    description: "Quote request type definition",

    inputs: {
        input_currency_type: {
            description: "Input currency type",
            type: "string",
            required: true
        },
        amount_requested: {
            description: "Requested amount in input currency",
            type: "string",
            required: true
        }
    },

    sync: true,

    fn: function (inputs, exits) {
        // your other validation logics
        return exits.success();
    }
});

, sync: true.

, :

inputs: {
    request: {
        type: "json",
        required: true,
        custom: require("../../types/QuoteRequest").validate
    }
},

, !

0

, , , :

, v1

./api/services ModelService.js :

module.exports = {
  invalidStructure(schema, input) {
    try {
      if (schema.type === 'array') {
        // Invalid: if input is not array
        //  OR some of input[items] doesn't match schema.item
        return !_.isArray(input) ||
          _.some(input, item => this.invalidStructure(schema.item, item));
      }
      else if (schema.type === 'object') {
        // Invalid if input is not an object
        //  OR if input.keys doesn't match schema.struct.keys
        //  OR if typeof input[key] doesn't match schema.struct[key]
        return !_.isObjectLike(input) ||
          !_.isEqual(_.keys(schema.struct), _.keys(input)) ||
          _.some(_.keys(input), key => this.invalidStructure(schema.struct[key], input[key]));
      }
      else { // verifying field value vs schema.type
        // TODO: Add other field validations here (i.e. isEmail, required,...etc.)
        return typeof input !== schema.type;
      }
    }
    catch (err) {
      sails.log.error('Exception in [invalidStructure] : ', err);
      return true;
    }
  }
}

:

const address = {
  type: 'object',
  struct: {
    name: { type: 'string' },
    location: {
      type: 'object',
      struct: {
        x: { type: 'string' },
        y: { type: 'string' },
        z: { type: 'string' },
      }
    }
  }
}
module.exports = {
  attributes: {
    address: {
      type: 'json',
      custom: value => !ModelService.invalidStructure(address, value)
    }
  }
}

, ( required: false)

json

Note. This is the source file . Feel free to use or improve with PR

0
source

All Articles