| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @module relations | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const _ = require('lodash/fp'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-05 18:35:16 +02:00
										 |  |  | const hasInversedBy = _.has('inversedBy'); | 
					
						
							|  |  |  | const hasMappedBy = _.has('mappedBy'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-12 16:12:40 +02:00
										 |  |  | const isOneToAny = attribute => ['oneToOne', 'oneToMany'].includes(attribute.relation); | 
					
						
							| 
									
										
										
										
											2021-07-05 18:35:16 +02:00
										 |  |  | const isBidirectional = attribute => hasInversedBy(attribute) || hasMappedBy(attribute); | 
					
						
							|  |  |  | const isOwner = attribute => !isBidirectional(attribute) || hasInversedBy(attribute); | 
					
						
							|  |  |  | const shouldUseJoinTable = attribute => attribute.useJoinTable !== false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Creates a oneToOne relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * if owner then | 
					
						
							|  |  |  |  *   if with join table then | 
					
						
							|  |  |  |  *     create join table | 
					
						
							|  |  |  |  *   else | 
					
						
							|  |  |  |  *     create joinColumn | 
					
						
							|  |  |  |  *   if bidirectional then | 
					
						
							|  |  |  |  *     set inverse attribute joinCol or joinTable info correctly | 
					
						
							|  |  |  |  * else | 
					
						
							|  |  |  |  *   this property must be set by the owner side | 
					
						
							|  |  |  |  *   verify the owner side is valid // should be done before or at the same time ?
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  * @retuns void | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const createOneToOne = (attributeName, attribute, meta, metadata) => { | 
					
						
							|  |  |  |   if (isOwner(attribute)) { | 
					
						
							|  |  |  |     if (shouldUseJoinTable(attribute)) { | 
					
						
							|  |  |  |       createJoinTable(metadata, { | 
					
						
							|  |  |  |         attribute, | 
					
						
							|  |  |  |         attributeName, | 
					
						
							|  |  |  |         meta, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       createJoinColum(metadata, { | 
					
						
							|  |  |  |         attribute, | 
					
						
							|  |  |  |         attributeName, | 
					
						
							|  |  |  |         meta, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     // TODO: verify other side is valid
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Creates a oneToMany relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * if unidirectional then | 
					
						
							|  |  |  |  *   create join table | 
					
						
							|  |  |  |  * if bidirectional then | 
					
						
							|  |  |  |  *   cannot be owning side | 
					
						
							|  |  |  |  *   do nothing | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const createOneToMany = (attributeName, attribute, meta, metadata) => { | 
					
						
							|  |  |  |   if (!isBidirectional(attribute)) { | 
					
						
							|  |  |  |     createJoinTable(metadata, { | 
					
						
							|  |  |  |       attribute, | 
					
						
							|  |  |  |       attributeName, | 
					
						
							|  |  |  |       meta, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (isOwner(attribute)) { | 
					
						
							|  |  |  |       throw new Error( | 
					
						
							|  |  |  |         'one side of a oneToMany cannot be the owner side in a bidirectional relation' | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Creates a manyToOne relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * if unidirectional then | 
					
						
							|  |  |  |  *   if with join table then | 
					
						
							|  |  |  |  *     create join table | 
					
						
							|  |  |  |  *   else | 
					
						
							|  |  |  |  *     create join column | 
					
						
							|  |  |  |  * else | 
					
						
							|  |  |  |  *   must be the owner side | 
					
						
							|  |  |  |  *   if with join table then | 
					
						
							|  |  |  |  *     create join table | 
					
						
							|  |  |  |  *   else | 
					
						
							|  |  |  |  *     create join column | 
					
						
							|  |  |  |  *   set inverse attribute joinCol or joinTable info correctly | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const createManyToOne = (attributeName, attribute, meta, metadata) => { | 
					
						
							|  |  |  |   if (isBidirectional(attribute) && !isOwner(attribute)) { | 
					
						
							|  |  |  |     throw new Error('The many side of a manyToOne must be the owning side'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (shouldUseJoinTable(attribute)) { | 
					
						
							|  |  |  |     createJoinTable(metadata, { | 
					
						
							|  |  |  |       attribute, | 
					
						
							|  |  |  |       attributeName, | 
					
						
							|  |  |  |       meta, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     createJoinColum(metadata, { | 
					
						
							|  |  |  |       attribute, | 
					
						
							|  |  |  |       attributeName, | 
					
						
							|  |  |  |       meta, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Creates a manyToMany relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * if unidirectional | 
					
						
							|  |  |  |  *   create join table | 
					
						
							|  |  |  |  * else | 
					
						
							|  |  |  |  *   if owner then | 
					
						
							|  |  |  |  *     if with join table then | 
					
						
							|  |  |  |  *       create join table | 
					
						
							|  |  |  |  *   else | 
					
						
							|  |  |  |  *     do nothing | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const createManyToMany = (attributeName, attribute, meta, metadata) => { | 
					
						
							|  |  |  |   if (!isBidirectional(attribute) || isOwner(attribute)) { | 
					
						
							|  |  |  |     createJoinTable(metadata, { | 
					
						
							|  |  |  |       attribute, | 
					
						
							|  |  |  |       attributeName, | 
					
						
							|  |  |  |       meta, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 17:52:59 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Creates a morphToOne relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * if with join table then | 
					
						
							|  |  |  |  *   create join table | 
					
						
							|  |  |  |  * else | 
					
						
							|  |  |  |  *  create join columnsa | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * if bidirectionnal | 
					
						
							|  |  |  |  *  set info in the traget | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-30 20:45:51 +02:00
										 |  |  | const createMorphToOne = (attributeName, attribute /*meta, metadata*/) => { | 
					
						
							| 
									
										
										
										
											2021-07-26 17:52:59 +02:00
										 |  |  |   const idColumnName = 'target_id'; | 
					
						
							|  |  |  |   const typeColumnName = 'target_type'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Object.assign(attribute, { | 
					
						
							|  |  |  |     owner: true, | 
					
						
							|  |  |  |     morphColumn: { | 
					
						
							|  |  |  |       // TODO: add referenced column
 | 
					
						
							|  |  |  |       typeColumn: { | 
					
						
							|  |  |  |         name: typeColumnName, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       idColumn: { | 
					
						
							|  |  |  |         name: idColumnName, | 
					
						
							|  |  |  |         referencedColumn: 'id', | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // TODO: implement bidirectional
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Creates a morphToMany relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const createMorphToMany = (attributeName, attribute, meta, metadata) => { | 
					
						
							|  |  |  |   const joinTableName = _.snakeCase(`${meta.tableName}_${attributeName}_morphs`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const joinColumnName = _.snakeCase(`${meta.singularName}_id`); | 
					
						
							|  |  |  |   const morphColumnName = _.snakeCase(`${attributeName}`); | 
					
						
							|  |  |  |   const idColumnName = `${morphColumnName}_id`; | 
					
						
							|  |  |  |   const typeColumnName = `${morphColumnName}_type`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   metadata.add({ | 
					
						
							|  |  |  |     uid: joinTableName, | 
					
						
							|  |  |  |     tableName: joinTableName, | 
					
						
							|  |  |  |     attributes: { | 
					
						
							|  |  |  |       [joinColumnName]: { | 
					
						
							|  |  |  |         type: 'integer', | 
					
						
							|  |  |  |         column: { | 
					
						
							|  |  |  |           unsigned: true, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       [idColumnName]: { | 
					
						
							|  |  |  |         type: 'integer', | 
					
						
							|  |  |  |         column: { | 
					
						
							|  |  |  |           unsigned: true, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       [typeColumnName]: { | 
					
						
							|  |  |  |         type: 'string', | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2021-08-17 18:08:42 +02:00
										 |  |  |       field: { | 
					
						
							|  |  |  |         type: 'string', | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2021-07-26 17:52:59 +02:00
										 |  |  |       order: { | 
					
						
							|  |  |  |         type: 'integer', | 
					
						
							|  |  |  |         column: { | 
					
						
							|  |  |  |           unsigned: true, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2021-09-16 11:37:44 +02:00
										 |  |  |     indexes: [ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         name: `${joinTableName}_fk`, | 
					
						
							|  |  |  |         columns: [joinColumnName], | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     ], | 
					
						
							| 
									
										
										
										
											2021-07-26 17:52:59 +02:00
										 |  |  |     foreignKeys: [ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         name: `${joinTableName}_fk`, | 
					
						
							|  |  |  |         columns: [joinColumnName], | 
					
						
							|  |  |  |         referencedColumns: ['id'], | 
					
						
							|  |  |  |         referencedTable: meta.tableName, | 
					
						
							|  |  |  |         onDelete: 'CASCADE', | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     ], | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const joinTable = { | 
					
						
							|  |  |  |     name: joinTableName, | 
					
						
							|  |  |  |     joinColumn: { | 
					
						
							|  |  |  |       name: joinColumnName, | 
					
						
							|  |  |  |       referencedColumn: 'id', | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     morphColumn: { | 
					
						
							|  |  |  |       typeColumn: { | 
					
						
							|  |  |  |         name: typeColumnName, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       idColumn: { | 
					
						
							|  |  |  |         name: idColumnName, | 
					
						
							|  |  |  |         referencedColumn: 'id', | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2021-08-17 18:08:42 +02:00
										 |  |  |     orderBy: { | 
					
						
							|  |  |  |       order: 'asc', | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2021-07-26 17:52:59 +02:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   attribute.joinTable = joinTable; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Creates a morphOne relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const createMorphOne = (attributeName, attribute, meta, metadata) => { | 
					
						
							|  |  |  |   const targetMeta = metadata.get(attribute.target); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!targetMeta) { | 
					
						
							|  |  |  |     throw new Error(`Morph target not found. Looking for ${attribute.target}`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!_.has(attribute.morphBy, targetMeta.attributes)) { | 
					
						
							|  |  |  |     throw new Error(`Morph target attribute not found. Looking for ${attribute.morphBy}`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Creates a morphMany relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const createMorphMany = (attributeName, attribute, meta, metadata) => { | 
					
						
							|  |  |  |   const targetMeta = metadata.get(attribute.target); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!targetMeta) { | 
					
						
							|  |  |  |     throw new Error(`Morph target not found. Looking for ${attribute.target}`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!_.has(attribute.morphBy, targetMeta.attributes)) { | 
					
						
							|  |  |  |     throw new Error(`Morph target attribute not found. Looking for ${attribute.morphBy}`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Creates a relation metadata | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {string} attributeName | 
					
						
							|  |  |  |  * @param {Attribute} attribute | 
					
						
							|  |  |  |  * @param {ModelMetadata} meta | 
					
						
							|  |  |  |  * @param {Metadata} metadata | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const createRelation = (attributeName, attribute, meta, metadata) => { | 
					
						
							| 
									
										
										
										
											2021-08-17 18:08:42 +02:00
										 |  |  |   switch (attribute.relation) { | 
					
						
							|  |  |  |     case 'oneToOne': | 
					
						
							|  |  |  |       return createOneToOne(attributeName, attribute, meta, metadata); | 
					
						
							|  |  |  |     case 'oneToMany': | 
					
						
							|  |  |  |       return createOneToMany(attributeName, attribute, meta, metadata); | 
					
						
							|  |  |  |     case 'manyToOne': | 
					
						
							|  |  |  |       return createManyToOne(attributeName, attribute, meta, metadata); | 
					
						
							|  |  |  |     case 'manyToMany': | 
					
						
							|  |  |  |       return createManyToMany(attributeName, attribute, meta, metadata); | 
					
						
							|  |  |  |     case 'morphToOne': | 
					
						
							|  |  |  |       return createMorphToOne(attributeName, attribute, meta, metadata); | 
					
						
							|  |  |  |     case 'morphToMany': | 
					
						
							|  |  |  |       return createMorphToMany(attributeName, attribute, meta, metadata); | 
					
						
							|  |  |  |     case 'morphOne': | 
					
						
							|  |  |  |       return createMorphOne(attributeName, attribute, meta, metadata); | 
					
						
							|  |  |  |     case 'morphMany': | 
					
						
							|  |  |  |       return createMorphMany(attributeName, attribute, meta, metadata); | 
					
						
							| 
									
										
										
										
											2021-07-26 17:52:59 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   throw new Error(`Unknown relation ${attribute.relation}`); | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-17 18:08:42 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Creates a join column info and add them to the attribute meta | 
					
						
							|  |  |  |  * @param {Object} metadata metadata registry | 
					
						
							|  |  |  |  * @param {Object} param | 
					
						
							|  |  |  |  * @param {Object} param.attribute associated attribute | 
					
						
							|  |  |  |  * @param {string} param.attributeName name of the associated attribute | 
					
						
							|  |  |  |  * @param {Object} param.meta model metadata | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-07 18:04:39 +02:00
										 |  |  | const createJoinColum = (metadata, { attribute, attributeName /*meta */ }) => { | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  |   const targetMeta = metadata.get(attribute.target); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-07 18:04:39 +02:00
										 |  |  |   const joinColumnName = _.snakeCase(`${attributeName}_id`); | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  |   const joinColumn = { | 
					
						
							|  |  |  |     name: joinColumnName, | 
					
						
							|  |  |  |     referencedColumn: 'id', | 
					
						
							|  |  |  |     referencedTable: targetMeta.tableName, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Object.assign(attribute, { owner: true, joinColumn }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (isBidirectional(attribute)) { | 
					
						
							|  |  |  |     const inverseAttribute = targetMeta.attributes[attribute.inversedBy]; | 
					
						
							| 
									
										
										
										
											2021-07-30 20:45:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  |     Object.assign(inverseAttribute, { | 
					
						
							|  |  |  |       joinColumn: { | 
					
						
							|  |  |  |         name: joinColumn.referencedColumn, | 
					
						
							|  |  |  |         referencedColumn: joinColumn.name, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-17 18:08:42 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Creates a join table and add it to the attribute meta | 
					
						
							|  |  |  |  * @param {Object} metadata metadata registry | 
					
						
							|  |  |  |  * @param {Object} param | 
					
						
							|  |  |  |  * @param {Object} param.attribute associated attribute | 
					
						
							|  |  |  |  * @param {string} param.attributeName name of the associated attribute | 
					
						
							|  |  |  |  * @param {Object} param.meta model metadata | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  | const createJoinTable = (metadata, { attributeName, attribute, meta }) => { | 
					
						
							|  |  |  |   const targetMeta = metadata.get(attribute.target); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!targetMeta) { | 
					
						
							|  |  |  |     throw new Error(`Unknow target ${attribute.target}`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const joinTableName = _.snakeCase(`${meta.tableName}_${attributeName}_links`); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-05 18:35:16 +02:00
										 |  |  |   let joinColumnName = _.snakeCase(`${meta.singularName}_id`); | 
					
						
							|  |  |  |   let inverseJoinColumnName = _.snakeCase(`${targetMeta.singularName}_id`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // if relation is slef referencing
 | 
					
						
							|  |  |  |   if (joinColumnName === inverseJoinColumnName) { | 
					
						
							|  |  |  |     inverseJoinColumnName = `inv_${inverseJoinColumnName}`; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   metadata.add({ | 
					
						
							|  |  |  |     uid: joinTableName, | 
					
						
							|  |  |  |     tableName: joinTableName, | 
					
						
							|  |  |  |     attributes: { | 
					
						
							|  |  |  |       [joinColumnName]: { | 
					
						
							|  |  |  |         type: 'integer', | 
					
						
							|  |  |  |         column: { | 
					
						
							|  |  |  |           unsigned: true, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       [inverseJoinColumnName]: { | 
					
						
							|  |  |  |         type: 'integer', | 
					
						
							|  |  |  |         column: { | 
					
						
							|  |  |  |           unsigned: true, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       // TODO: add extra pivot attributes -> user should use an intermediate entity
 | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2021-09-16 11:37:44 +02:00
										 |  |  |     indexes: [ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         name: `${joinTableName}_fk`, | 
					
						
							|  |  |  |         columns: [joinColumnName], | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         name: `${joinTableName}_inv_fk`, | 
					
						
							|  |  |  |         columns: [inverseJoinColumnName], | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     ], | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  |     foreignKeys: [ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         name: `${joinTableName}_fk`, | 
					
						
							|  |  |  |         columns: [joinColumnName], | 
					
						
							|  |  |  |         referencedColumns: ['id'], | 
					
						
							|  |  |  |         referencedTable: meta.tableName, | 
					
						
							|  |  |  |         onDelete: 'CASCADE', | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         name: `${joinTableName}_inv_fk`, | 
					
						
							|  |  |  |         columns: [inverseJoinColumnName], | 
					
						
							|  |  |  |         referencedColumns: ['id'], | 
					
						
							|  |  |  |         referencedTable: targetMeta.tableName, | 
					
						
							|  |  |  |         onDelete: 'CASCADE', | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     ], | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const joinTable = { | 
					
						
							|  |  |  |     name: joinTableName, | 
					
						
							|  |  |  |     joinColumn: { | 
					
						
							|  |  |  |       name: joinColumnName, | 
					
						
							|  |  |  |       referencedColumn: 'id', | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     inverseJoinColumn: { | 
					
						
							|  |  |  |       name: inverseJoinColumnName, | 
					
						
							|  |  |  |       referencedColumn: 'id', | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   attribute.joinTable = joinTable; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (isBidirectional(attribute)) { | 
					
						
							|  |  |  |     const inverseAttribute = targetMeta.attributes[attribute.inversedBy]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!inverseAttribute) { | 
					
						
							| 
									
										
										
										
											2021-07-08 18:15:32 +02:00
										 |  |  |       throw new Error( | 
					
						
							|  |  |  |         `inversedBy attribute ${attribute.inversedBy} not found target ${targetMeta.uid}` | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-02 02:26:14 +02:00
										 |  |  |     inverseAttribute.joinTable = { | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  |       name: joinTableName, | 
					
						
							|  |  |  |       joinColumn: joinTable.inverseJoinColumn, | 
					
						
							|  |  |  |       inverseJoinColumn: joinTable.joinColumn, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = { | 
					
						
							|  |  |  |   createRelation, | 
					
						
							| 
									
										
										
										
											2021-07-05 18:35:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   isBidirectional, | 
					
						
							| 
									
										
										
										
											2021-08-12 16:12:40 +02:00
										 |  |  |   isOneToAny, | 
					
						
							| 
									
										
										
										
											2021-07-01 14:32:50 +02:00
										 |  |  | }; |