'use strict'; /** * Module dependencies */ // Node.js core. const fs = require('fs'); const path = require('path'); // Public node modules. const _ = require('lodash'); const pluralize = require('pluralize'); // Fetch stub attribute template on initial load. const attributeTemplate = fs.readFileSync(path.resolve(__dirname, '..', 'templates', 'attribute.template'), 'utf8'); /** * This `before` function is run before generating targets. * Validate, configure defaults, get extra dependencies, etc. * * @param {Object} scope * @param {Function} cb */ module.exports = (scope, cb) => { if (!scope.rootPath || !scope.id) { return cb.invalid('Usage: `$ strapi generate:model modelName --api apiName --plugin pluginName`'); } // `scope.args` are the raw command line arguments. _.defaults(scope, { id: _.trim(_.deburr(scope.id)), idPluralized: pluralize.plural(_.trim(_.deburr(scope.id))), environment: process.NODE_ENV || 'development' }); // Determine default values based on the available scope. _.defaults(scope, { globalID: _.upperFirst(_.camelCase(scope.id)), ext: '.js' }); // Determine the destination path. let filePath; if (scope.args.api) { filePath = `./api/${scope.args.api}/models`; } else if (scope.args.plugin) { filePath = `./plugins/${scope.args.plugin}/models`; } else { filePath = `./api/${scope.id}`; } // Take another pass to take advantage of the defaults absorbed in previous passes. _.defaults(scope, { rootPath: scope.rootPath, filePath, filename: scope.globalID + scope.ext, filenameSettings: scope.globalID + '.settings.json' }); // Humanize output. _.defaults(scope, { humanizeId: _.camelCase(scope.id).toLowerCase(), humanizedPath: '`' + scope.filePath + '`' }); // Validate optional attribute arguments. const invalidAttributes = []; // Map attributes and split them. scope.attributes = scope.args.attributes.map((attribute) => { const parts = attribute.split(':'); parts[1] = parts[1] ? parts[1] : 'string'; // Handle invalid attributes. if (!parts[1] || !parts[0]) { invalidAttributes.push('Error: Invalid attribute notation `' + attribute + '`.'); return; } return { name: _.trim(_.deburr(_.camelCase(parts[0]).toLowerCase())), type: _.trim(_.deburr(_.camelCase(parts[1]).toLowerCase())) }; }); // Handle invalid action arguments. // Send back invalidActions. if (invalidAttributes.length) { return cb.invalid(invalidAttributes); } // Make sure there aren't duplicates. if (_(scope.attributes.map(attribute => (attribute.name))).uniq().valueOf().length !== scope.attributes.length) { return cb.invalid('Duplicate attributes not allowed!'); } // Render some stringified code from the action template // and make it available in our scope for use later on. scope.attributes = scope.attributes.map((attribute) => { const compiled = _.template(attributeTemplate); return _.trimEnd(_.unescape(compiled({ name: attribute.name, type: attribute.type }))); }).join(',\n'); // Get default connection try { scope.connection = JSON.parse(fs.readFileSync(path.resolve(scope.rootPath, 'config', 'environments', scope.environment, 'database.json'))).defaultConnection || ''; } catch (err) { return cb.invalid(err); } // Trigger callback with no error to proceed. return cb(); };