123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631 |
- 'use strict';
- const DataTypes = require('./data-types');
- const SqlString = require('./sql-string');
- const _ = require('lodash');
- const uuidv1 = require('uuid').v1;
- const uuidv4 = require('uuid').v4;
- const operators = require('./operators');
- const operatorsSet = new Set(Object.values(operators));
- let inflection = require('inflection');
- exports.classToInvokable = require('./utils/class-to-invokable').classToInvokable;
- exports.joinSQLFragments = require('./utils/join-sql-fragments').joinSQLFragments;
- function useInflection(_inflection) {
- inflection = _inflection;
- }
- exports.useInflection = useInflection;
- function camelizeIf(str, condition) {
- let result = str;
- if (condition) {
- result = camelize(str);
- }
- return result;
- }
- exports.camelizeIf = camelizeIf;
- function underscoredIf(str, condition) {
- let result = str;
- if (condition) {
- result = underscore(str);
- }
- return result;
- }
- exports.underscoredIf = underscoredIf;
- function isPrimitive(val) {
- const type = typeof val;
- return type === 'string' || type === 'number' || type === 'boolean';
- }
- exports.isPrimitive = isPrimitive;
- // Same concept as _.merge, but don't overwrite properties that have already been assigned
- function mergeDefaults(a, b) {
- return _.mergeWith(a, b, (objectValue, sourceValue) => {
- // If it's an object, let _ handle it this time, we will be called again for each property
- if (!_.isPlainObject(objectValue) && objectValue !== undefined) {
- if (_.isFunction(objectValue) && _.isNative(objectValue)) {
- return sourceValue || objectValue;
- }
- return objectValue;
- }
- });
- }
- exports.mergeDefaults = mergeDefaults;
- // An alternative to _.merge, which doesn't clone its arguments
- // Cloning is a bad idea because options arguments may contain references to sequelize
- // models - which again reference database libs which don't like to be cloned (in particular pg-native)
- function merge() {
- const result = {};
- for (const obj of arguments) {
- _.forOwn(obj, (value, key) => {
- if (value !== undefined) {
- if (!result[key]) {
- result[key] = value;
- } else if (_.isPlainObject(value) && _.isPlainObject(result[key])) {
- result[key] = merge(result[key], value);
- } else if (Array.isArray(value) && Array.isArray(result[key])) {
- result[key] = value.concat(result[key]);
- } else {
- result[key] = value;
- }
- }
- });
- }
- return result;
- }
- exports.merge = merge;
- function spliceStr(str, index, count, add) {
- return str.slice(0, index) + add + str.slice(index + count);
- }
- exports.spliceStr = spliceStr;
- function camelize(str) {
- return str.trim().replace(/[-_\s]+(.)?/g, (match, c) => c.toUpperCase());
- }
- exports.camelize = camelize;
- function underscore(str) {
- return inflection.underscore(str);
- }
- exports.underscore = underscore;
- function singularize(str) {
- return inflection.singularize(str);
- }
- exports.singularize = singularize;
- function pluralize(str) {
- return inflection.pluralize(str);
- }
- exports.pluralize = pluralize;
- function format(arr, dialect) {
- const timeZone = null;
- // Make a clone of the array beacuse format modifies the passed args
- return SqlString.format(arr[0], arr.slice(1), timeZone, dialect);
- }
- exports.format = format;
- function formatNamedParameters(sql, parameters, dialect) {
- const timeZone = null;
- return SqlString.formatNamedParameters(sql, parameters, timeZone, dialect);
- }
- exports.formatNamedParameters = formatNamedParameters;
- function cloneDeep(obj, onlyPlain) {
- obj = obj || {};
- return _.cloneDeepWith(obj, elem => {
- // Do not try to customize cloning of arrays or POJOs
- if (Array.isArray(elem) || _.isPlainObject(elem)) {
- return undefined;
- }
- // If we specified to clone only plain objects & arrays, we ignore everyhing else
- // In any case, don't clone stuff that's an object, but not a plain one - fx example sequelize models and instances
- if (onlyPlain || typeof elem === 'object') {
- return elem;
- }
- // Preserve special data-types like `fn` across clones. _.get() is used for checking up the prototype chain
- if (elem && typeof elem.clone === 'function') {
- return elem.clone();
- }
- });
- }
- exports.cloneDeep = cloneDeep;
- /* Expand and normalize finder options */
- function mapFinderOptions(options, Model) {
- if (options.attributes && Array.isArray(options.attributes)) {
- options.attributes = Model._injectDependentVirtualAttributes(options.attributes);
- options.attributes = options.attributes.filter(v => !Model._virtualAttributes.has(v));
- }
- mapOptionFieldNames(options, Model);
- return options;
- }
- exports.mapFinderOptions = mapFinderOptions;
- /* Used to map field names in attributes and where conditions */
- function mapOptionFieldNames(options, Model) {
- if (Array.isArray(options.attributes)) {
- options.attributes = options.attributes.map(attr => {
- // Object lookups will force any variable to strings, we don't want that for special objects etc
- if (typeof attr !== 'string') return attr;
- // Map attributes to aliased syntax attributes
- if (Model.rawAttributes[attr] && attr !== Model.rawAttributes[attr].field) {
- return [Model.rawAttributes[attr].field, attr];
- }
- return attr;
- });
- }
- if (options.where && _.isPlainObject(options.where)) {
- options.where = mapWhereFieldNames(options.where, Model);
- }
- return options;
- }
- exports.mapOptionFieldNames = mapOptionFieldNames;
- function mapWhereFieldNames(attributes, Model) {
- if (attributes) {
- getComplexKeys(attributes).forEach(attribute => {
- const rawAttribute = Model.rawAttributes[attribute];
- if (rawAttribute && rawAttribute.field !== rawAttribute.fieldName) {
- attributes[rawAttribute.field] = attributes[attribute];
- delete attributes[attribute];
- }
- if (_.isPlainObject(attributes[attribute])
- && !(rawAttribute && (
- rawAttribute.type instanceof DataTypes.HSTORE
- || rawAttribute.type instanceof DataTypes.JSON))) { // Prevent renaming of HSTORE & JSON fields
- attributes[attribute] = mapOptionFieldNames({
- where: attributes[attribute]
- }, Model).where;
- }
- if (Array.isArray(attributes[attribute])) {
- attributes[attribute].forEach((where, index) => {
- if (_.isPlainObject(where)) {
- attributes[attribute][index] = mapWhereFieldNames(where, Model);
- }
- });
- }
- });
- }
- return attributes;
- }
- exports.mapWhereFieldNames = mapWhereFieldNames;
- /* Used to map field names in values */
- function mapValueFieldNames(dataValues, fields, Model) {
- const values = {};
- for (const attr of fields) {
- if (dataValues[attr] !== undefined && !Model._virtualAttributes.has(attr)) {
- // Field name mapping
- if (Model.rawAttributes[attr] && Model.rawAttributes[attr].field && Model.rawAttributes[attr].field !== attr) {
- values[Model.rawAttributes[attr].field] = dataValues[attr];
- } else {
- values[attr] = dataValues[attr];
- }
- }
- }
- return values;
- }
- exports.mapValueFieldNames = mapValueFieldNames;
- function isColString(value) {
- return typeof value === 'string' && value[0] === '$' && value[value.length - 1] === '$';
- }
- exports.isColString = isColString;
- function canTreatArrayAsAnd(arr) {
- return arr.some(arg => _.isPlainObject(arg) || arg instanceof Where);
- }
- exports.canTreatArrayAsAnd = canTreatArrayAsAnd;
- function combineTableNames(tableName1, tableName2) {
- return tableName1.toLowerCase() < tableName2.toLowerCase() ? tableName1 + tableName2 : tableName2 + tableName1;
- }
- exports.combineTableNames = combineTableNames;
- function toDefaultValue(value, dialect) {
- if (typeof value === 'function') {
- const tmp = value();
- if (tmp instanceof DataTypes.ABSTRACT) {
- return tmp.toSql();
- }
- return tmp;
- }
- if (value instanceof DataTypes.UUIDV1) {
- return uuidv1();
- }
- if (value instanceof DataTypes.UUIDV4) {
- return uuidv4();
- }
- if (value instanceof DataTypes.NOW) {
- return now(dialect);
- }
- if (Array.isArray(value)) {
- return value.slice();
- }
- if (_.isPlainObject(value)) {
- return { ...value };
- }
- return value;
- }
- exports.toDefaultValue = toDefaultValue;
- /**
- * Determine if the default value provided exists and can be described
- * in a db schema using the DEFAULT directive.
- *
- * @param {*} value Any default value.
- * @returns {boolean} yes / no.
- * @private
- */
- function defaultValueSchemable(value) {
- if (value === undefined) { return false; }
- // TODO this will be schemable when all supported db
- // have been normalized for this case
- if (value instanceof DataTypes.NOW) { return false; }
- if (value instanceof DataTypes.UUIDV1 || value instanceof DataTypes.UUIDV4) { return false; }
- return typeof value !== 'function';
- }
- exports.defaultValueSchemable = defaultValueSchemable;
- function removeNullValuesFromHash(hash, omitNull, options) {
- let result = hash;
- options = options || {};
- options.allowNull = options.allowNull || [];
- if (omitNull) {
- const _hash = {};
- _.forIn(hash, (val, key) => {
- if (options.allowNull.includes(key) || key.endsWith('Id') || val !== null && val !== undefined) {
- _hash[key] = val;
- }
- });
- result = _hash;
- }
- return result;
- }
- exports.removeNullValuesFromHash = removeNullValuesFromHash;
- const dialects = new Set(['mariadb', 'mysql', 'postgres', 'sqlite', 'mssql']);
- function now(dialect) {
- const d = new Date();
- if (!dialects.has(dialect)) {
- d.setMilliseconds(0);
- }
- return d;
- }
- exports.now = now;
- // Note: Use the `quoteIdentifier()` and `escape()` methods on the
- // `QueryInterface` instead for more portable code.
- const TICK_CHAR = '`';
- exports.TICK_CHAR = TICK_CHAR;
- function addTicks(s, tickChar) {
- tickChar = tickChar || TICK_CHAR;
- return tickChar + removeTicks(s, tickChar) + tickChar;
- }
- exports.addTicks = addTicks;
- function removeTicks(s, tickChar) {
- tickChar = tickChar || TICK_CHAR;
- return s.replace(new RegExp(tickChar, 'g'), '');
- }
- exports.removeTicks = removeTicks;
- /**
- * Receives a tree-like object and returns a plain object which depth is 1.
- *
- * - Input:
- *
- * {
- * name: 'John',
- * address: {
- * street: 'Fake St. 123',
- * coordinates: {
- * longitude: 55.6779627,
- * latitude: 12.5964313
- * }
- * }
- * }
- *
- * - Output:
- *
- * {
- * name: 'John',
- * address.street: 'Fake St. 123',
- * address.coordinates.latitude: 55.6779627,
- * address.coordinates.longitude: 12.5964313
- * }
- *
- * @param {object} value an Object
- * @returns {object} a flattened object
- * @private
- */
- function flattenObjectDeep(value) {
- if (!_.isPlainObject(value)) return value;
- const flattenedObj = {};
- function flattenObject(obj, subPath) {
- Object.keys(obj).forEach(key => {
- const pathToProperty = subPath ? `${subPath}.${key}` : key;
- if (typeof obj[key] === 'object' && obj[key] !== null) {
- flattenObject(obj[key], pathToProperty);
- } else {
- flattenedObj[pathToProperty] = _.get(obj, key);
- }
- });
- return flattenedObj;
- }
- return flattenObject(value, undefined);
- }
- exports.flattenObjectDeep = flattenObjectDeep;
- /**
- * Utility functions for representing SQL functions, and columns that should be escaped.
- * Please do not use these functions directly, use Sequelize.fn and Sequelize.col instead.
- *
- * @private
- */
- class SequelizeMethod {}
- exports.SequelizeMethod = SequelizeMethod;
- class Fn extends SequelizeMethod {
- constructor(fn, args) {
- super();
- this.fn = fn;
- this.args = args;
- }
- clone() {
- return new Fn(this.fn, this.args);
- }
- }
- exports.Fn = Fn;
- class Col extends SequelizeMethod {
- constructor(col, ...args) {
- super();
- if (args.length > 0) {
- col = args;
- }
- this.col = col;
- }
- }
- exports.Col = Col;
- class Cast extends SequelizeMethod {
- constructor(val, type, json) {
- super();
- this.val = val;
- this.type = (type || '').trim();
- this.json = json || false;
- }
- }
- exports.Cast = Cast;
- class Literal extends SequelizeMethod {
- constructor(val) {
- super();
- this.val = val;
- }
- }
- exports.Literal = Literal;
- class Json extends SequelizeMethod {
- constructor(conditionsOrPath, value) {
- super();
- if (_.isObject(conditionsOrPath)) {
- this.conditions = conditionsOrPath;
- } else {
- this.path = conditionsOrPath;
- if (value) {
- this.value = value;
- }
- }
- }
- }
- exports.Json = Json;
- class Where extends SequelizeMethod {
- constructor(attribute, comparator, logic) {
- super();
- if (logic === undefined) {
- logic = comparator;
- comparator = '=';
- }
- this.attribute = attribute;
- this.comparator = comparator;
- this.logic = logic;
- }
- }
- exports.Where = Where;
- //Collection of helper methods to make it easier to work with symbol operators
- /**
- * getOperators
- *
- * @param {object} obj
- * @returns {Array<symbol>} All operators properties of obj
- * @private
- */
- function getOperators(obj) {
- return Object.getOwnPropertySymbols(obj).filter(s => operatorsSet.has(s));
- }
- exports.getOperators = getOperators;
- /**
- * getComplexKeys
- *
- * @param {object} obj
- * @returns {Array<string|symbol>} All keys including operators
- * @private
- */
- function getComplexKeys(obj) {
- return getOperators(obj).concat(Object.keys(obj));
- }
- exports.getComplexKeys = getComplexKeys;
- /**
- * getComplexSize
- *
- * @param {object|Array} obj
- * @returns {number} Length of object properties including operators if obj is array returns its length
- * @private
- */
- function getComplexSize(obj) {
- return Array.isArray(obj) ? obj.length : getComplexKeys(obj).length;
- }
- exports.getComplexSize = getComplexSize;
- /**
- * Returns true if a where clause is empty, even with Symbols
- *
- * @param {object} obj
- * @returns {boolean}
- * @private
- */
- function isWhereEmpty(obj) {
- return !!obj && _.isEmpty(obj) && getOperators(obj).length === 0;
- }
- exports.isWhereEmpty = isWhereEmpty;
- /**
- * Returns ENUM name by joining table and column name
- *
- * @param {string} tableName
- * @param {string} columnName
- * @returns {string}
- * @private
- */
- function generateEnumName(tableName, columnName) {
- return `enum_${tableName}_${columnName}`;
- }
- exports.generateEnumName = generateEnumName;
- /**
- * Returns an new Object which keys are camelized
- *
- * @param {object} obj
- * @returns {string}
- * @private
- */
- function camelizeObjectKeys(obj) {
- const newObj = new Object();
- Object.keys(obj).forEach(key => {
- newObj[camelize(key)] = obj[key];
- });
- return newObj;
- }
- exports.camelizeObjectKeys = camelizeObjectKeys;
- /**
- * Assigns own and inherited enumerable string and symbol keyed properties of source
- * objects to the destination object.
- *
- * https://lodash.com/docs/4.17.4#defaults
- *
- * **Note:** This method mutates `object`.
- *
- * @param {object} object The destination object.
- * @param {...object} [sources] The source objects.
- * @returns {object} Returns `object`.
- * @private
- */
- function defaults(object, ...sources) {
- object = Object(object);
- sources.forEach(source => {
- if (source) {
- source = Object(source);
- getComplexKeys(source).forEach(key => {
- const value = object[key];
- if (
- value === undefined ||
- _.eq(value, Object.prototype[key]) &&
- !Object.prototype.hasOwnProperty.call(object, key)
- ) {
- object[key] = source[key];
- }
- });
- }
- });
- return object;
- }
- exports.defaults = defaults;
- /**
- *
- * @param {object} index
- * @param {Array} index.fields
- * @param {string} [index.name]
- * @param {string|object} tableName
- *
- * @returns {object}
- * @private
- */
- function nameIndex(index, tableName) {
- if (tableName.tableName) tableName = tableName.tableName;
- if (!Object.prototype.hasOwnProperty.call(index, 'name')) {
- const fields = index.fields.map(
- field => typeof field === 'string' ? field : field.name || field.attribute
- );
- index.name = underscore(`${tableName}_${fields.join('_')}`);
- }
- return index;
- }
- exports.nameIndex = nameIndex;
- /**
- * Checks if 2 arrays intersect.
- *
- * @param {Array} arr1
- * @param {Array} arr2
- * @private
- */
- function intersects(arr1, arr2) {
- return arr1.some(v => arr2.includes(v));
- }
- exports.intersects = intersects;
|