1257 lines
33 KiB
JavaScript
1257 lines
33 KiB
JavaScript
exports.up = knex => Promise.resolve()
|
|
.then(() => knex.schema.createTable('countries', (table) => {
|
|
table.text('alpha2', 2)
|
|
.unique()
|
|
.primary();
|
|
|
|
table.text('alpha3', 3)
|
|
.unique();
|
|
|
|
table.text('name')
|
|
.notNullable();
|
|
|
|
table.text('alias');
|
|
|
|
table.integer('code', 3);
|
|
table.text('nationality');
|
|
|
|
table.integer('priority', 2)
|
|
.defaultTo(0);
|
|
}))
|
|
.then(() => knex.schema.createTable('media', (table) => {
|
|
table.text('id', 21)
|
|
.primary();
|
|
|
|
table.text('path');
|
|
table.text('thumbnail');
|
|
table.text('lazy');
|
|
table.integer('index');
|
|
table.text('mime');
|
|
|
|
table.text('hash');
|
|
|
|
table.integer('size', 12);
|
|
table.integer('quality', 6);
|
|
table.integer('width', 6);
|
|
table.integer('height', 6);
|
|
table.float('entropy');
|
|
|
|
table.text('scraper', 32);
|
|
table.text('credit', 100);
|
|
|
|
table.text('source', 2100);
|
|
table.text('source_page', 2100);
|
|
|
|
table.text('comment');
|
|
table.text('group');
|
|
|
|
table.boolean('is_sfw')
|
|
.defaultTo(false);
|
|
|
|
table.unique('hash');
|
|
table.unique('source');
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => { // eslint-disable-line arrow-body-style
|
|
// allow vim fold
|
|
return knex.raw(`
|
|
CREATE FUNCTION get_random_sfw_media_id() RETURNS varchar AS $$
|
|
SELECT id FROM media
|
|
WHERE is_sfw = true
|
|
ORDER BY random()
|
|
LIMIT 1;
|
|
$$ LANGUAGE sql STABLE;
|
|
`);
|
|
})
|
|
.then(() => knex.schema.alterTable('media', (table) => {
|
|
table.text('sfw_media_id', 21)
|
|
.references('id')
|
|
.inTable('media')
|
|
.defaultTo(knex.raw('get_random_sfw_media_id()'));
|
|
}))
|
|
.then(() => knex.schema.createTable('tags_groups', (table) => {
|
|
table.increments('id', 12);
|
|
|
|
table.text('name', 32);
|
|
table.text('description');
|
|
|
|
table.text('slug', 32)
|
|
.unique();
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('tags', (table) => {
|
|
table.increments('id', 12);
|
|
table.text('name');
|
|
|
|
table.text('description');
|
|
|
|
table.integer('priority', 2)
|
|
.defaultTo(0);
|
|
|
|
table.boolean('filter')
|
|
.defaultTo(false);
|
|
|
|
table.boolean('secondary')
|
|
.defaultTo(false);
|
|
|
|
table.integer('group_id', 12)
|
|
.references('id')
|
|
.inTable('tags_groups');
|
|
|
|
table.integer('alias_for', 12)
|
|
.references('id')
|
|
.inTable('tags');
|
|
|
|
table.text('slug', 32)
|
|
.unique();
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('tags_posters', (table) => {
|
|
table.integer('tag_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('tags');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique('tag_id');
|
|
}))
|
|
.then(() => knex.schema.createTable('tags_photos', (table) => {
|
|
table.integer('tag_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('tags');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique(['tag_id', 'media_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('entities_types', (table) => {
|
|
table.text('type')
|
|
.primary();
|
|
}))
|
|
.then(() => { // eslint-disable-line arrow-body-style
|
|
// allow vim fold
|
|
return knex('entities_types').insert([
|
|
{ type: 'network' },
|
|
{ type: 'channel' },
|
|
{ type: 'studio' },
|
|
{ type: 'info' },
|
|
]);
|
|
})
|
|
.then(() => knex.schema.createTable('entities', (table) => {
|
|
table.increments('id', 12);
|
|
|
|
table.integer('parent_id', 12)
|
|
.references('id')
|
|
.inTable('entities')
|
|
.index();
|
|
|
|
table.text('name');
|
|
table.text('slug', 32);
|
|
|
|
table.text('type')
|
|
.notNullable()
|
|
.references('type')
|
|
.inTable('entities_types')
|
|
.defaultTo('channel');
|
|
|
|
table.unique(['slug', 'type']);
|
|
|
|
table.specificType('alias', 'text[]');
|
|
|
|
table.text('url');
|
|
table.text('description');
|
|
table.json('parameters');
|
|
|
|
table.integer('priority', 3)
|
|
.defaultTo(0);
|
|
|
|
table.boolean('independent')
|
|
.defaultTo(false);
|
|
|
|
table.boolean('visible')
|
|
.defaultTo(true);
|
|
|
|
table.boolean('has_logo')
|
|
.defaultTo(true);
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('entities_tags', (table) => {
|
|
table.integer('tag_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('tags');
|
|
|
|
table.integer('entity_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('entities');
|
|
|
|
table.boolean('inherit')
|
|
.defaultTo(false);
|
|
|
|
table.unique(['tag_id', 'entity_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('entities_social', (table) => {
|
|
table.increments('id', 16);
|
|
|
|
table.text('url');
|
|
table.text('platform');
|
|
|
|
table.integer('entity_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('entities');
|
|
|
|
table.unique(['url', 'entity_id']);
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('batches', (table) => {
|
|
table.increments('id', 12);
|
|
table.text('comment');
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('actors', (table) => {
|
|
table.increments('id', 12);
|
|
|
|
table.text('name')
|
|
.notNullable();
|
|
|
|
table.text('slug', 32)
|
|
.notNullable();
|
|
|
|
table.text('real_name');
|
|
|
|
table.integer('entity_id', 12)
|
|
.references('id')
|
|
.inTable('entities');
|
|
|
|
table.text('entry_id');
|
|
|
|
table.integer('alias_for', 12)
|
|
.references('id')
|
|
.inTable('actors');
|
|
|
|
table.date('date_of_birth');
|
|
table.date('date_of_death');
|
|
table.integer('age', 3);
|
|
|
|
table.text('gender', 18);
|
|
table.text('description');
|
|
|
|
table.text('birth_city');
|
|
table.text('birth_state');
|
|
table.text('birth_country_alpha2', 2)
|
|
.references('alpha2')
|
|
.inTable('countries');
|
|
|
|
table.text('residence_city');
|
|
table.text('residence_state');
|
|
table.text('residence_country_alpha2', 2)
|
|
.references('alpha2')
|
|
.inTable('countries');
|
|
|
|
table.text('ethnicity');
|
|
|
|
table.text('cup', 4);
|
|
table.integer('bust', 3);
|
|
table.integer('waist', 3);
|
|
table.integer('hip', 3);
|
|
table.boolean('natural_boobs');
|
|
|
|
table.integer('penis_length', 3);
|
|
table.integer('penis_girth', 3);
|
|
table.boolean('circumcised');
|
|
|
|
table.integer('height', 3);
|
|
table.integer('weight', 3);
|
|
table.text('eyes');
|
|
table.text('hair_color');
|
|
table.text('hair_length');
|
|
|
|
table.boolean('has_tattoos');
|
|
table.boolean('has_piercings');
|
|
table.text('piercings');
|
|
table.text('tattoos');
|
|
|
|
table.text('avatar_media_id', 21)
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.integer('batch_id', 12)
|
|
.references('id')
|
|
.inTable('batches')
|
|
.onDelete('cascade');
|
|
|
|
table.datetime('updated_at')
|
|
.defaultTo(knex.fn.now());
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('actors_profiles', (table) => {
|
|
table.increments('id', 12);
|
|
|
|
table.integer('actor_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('actors')
|
|
.onDelete('cascade');
|
|
|
|
table.integer('entity_id', 12)
|
|
.references('id')
|
|
.inTable('entities');
|
|
|
|
table.unique(['actor_id', 'entity_id']);
|
|
table.integer('priority', 4)
|
|
.defaultTo(1);
|
|
|
|
table.text('url');
|
|
|
|
table.text('real_name');
|
|
table.text('gender', 18);
|
|
|
|
table.date('date_of_birth');
|
|
table.date('date_of_death');
|
|
table.integer('age', 3);
|
|
|
|
table.text('description');
|
|
table.text('description_hash');
|
|
|
|
table.text('birth_city');
|
|
table.text('birth_state');
|
|
table.text('birth_country_alpha2', 2)
|
|
.references('alpha2')
|
|
.inTable('countries');
|
|
|
|
table.text('residence_city');
|
|
table.text('residence_state');
|
|
table.text('residence_country_alpha2', 2)
|
|
.references('alpha2')
|
|
.inTable('countries');
|
|
|
|
table.text('ethnicity');
|
|
|
|
table.text('cup', 4);
|
|
table.integer('bust', 3);
|
|
table.integer('waist', 3);
|
|
table.integer('hip', 3);
|
|
table.boolean('natural_boobs');
|
|
|
|
table.integer('penis_length', 3);
|
|
table.integer('penis_girth', 3);
|
|
table.boolean('circumcised');
|
|
|
|
table.integer('height', 3);
|
|
table.integer('weight', 3);
|
|
table.text('eyes');
|
|
table.text('hair_color');
|
|
table.text('hair_length');
|
|
|
|
table.boolean('has_tattoos');
|
|
table.boolean('has_piercings');
|
|
table.text('piercings');
|
|
table.text('tattoos');
|
|
|
|
table.text('avatar_media_id', 21)
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.datetime('scraped_at');
|
|
table.boolean('scrape_success');
|
|
|
|
table.datetime('updated_at')
|
|
.defaultTo(knex.fn.now());
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('body', (table) => {
|
|
table.text('slug', 20)
|
|
.primary();
|
|
|
|
table.text('name');
|
|
}))
|
|
.then(() => { // eslint-disable-line arrow-body-style
|
|
// allow vim fold
|
|
return knex('body').insert([
|
|
// head
|
|
{ slug: 'head', name: 'head' },
|
|
{ slug: 'face', name: 'face' },
|
|
{ slug: 'scalp', name: 'scalp' },
|
|
{ slug: 'forehead', name: 'forehead' },
|
|
{ slug: 'temple', name: 'temple' },
|
|
{ slug: 'cheek', name: 'cheek' },
|
|
{ slug: 'jaw', name: 'jaw' },
|
|
{ slug: 'chin', name: 'chin' },
|
|
{ slug: 'neck', name: 'neck' },
|
|
{ slug: 'throat', name: 'throat' },
|
|
// eyes
|
|
{ slug: 'eyelid', name: 'eyelid' },
|
|
{ slug: 'eyeball', name: 'eyeball' },
|
|
{ slug: 'eyebrow', name: 'eyebrow' },
|
|
// mouth
|
|
{ slug: 'tongue', name: 'tongue' },
|
|
{ slug: 'lip', name: 'lip' },
|
|
{ slug: 'upper-lip', name: 'upper lip' },
|
|
{ slug: 'lower-lip', name: 'lower lip' },
|
|
{ slug: 'inner-lip', name: 'inner lip' },
|
|
{ slug: 'inner-lower-lip', name: 'inner lower lip' },
|
|
{ slug: 'inner-upper-lip', name: 'inner upper lip' },
|
|
{ slug: 'philtrum', name: 'philtrum' },
|
|
{ slug: 'above-lip', name: 'above lip' },
|
|
{ slug: 'below-lip', name: 'below lip' },
|
|
// nose
|
|
{ slug: 'nose', name: 'nose' },
|
|
{ slug: 'third-eye', name: 'third eye' },
|
|
{ slug: 'bridge', name: 'bridge' },
|
|
{ slug: 'nostril', name: 'nostril' },
|
|
{ slug: 'septum', name: 'septum' },
|
|
{ slug: 'septril', name: 'septril' },
|
|
// ear
|
|
{ slug: 'ear', name: 'ear' },
|
|
{ slug: 'earlobe', name: 'earlobe' },
|
|
{ slug: 'helix', name: 'helix' },
|
|
{ slug: 'tragus', name: 'tragus' },
|
|
{ slug: 'conch', name: 'conch' },
|
|
{ slug: 'rook', name: 'rook' },
|
|
{ slug: 'behind-ear', name: 'behind ear' },
|
|
// arms
|
|
{ slug: 'arm', name: 'arm' },
|
|
{ slug: 'upper-arm', name: 'upper arm' },
|
|
{ slug: 'forearm', name: 'forearm' },
|
|
{ slug: 'elbow', name: 'elbow' },
|
|
{ slug: 'inner-elbow', name: 'inner elbow' },
|
|
{ slug: 'outer-elbow', name: 'outer elbow' },
|
|
// hands
|
|
{ slug: 'hand', name: 'hand' },
|
|
{ slug: 'fingers', name: 'fingers' },
|
|
{ slug: 'knuckles', name: 'knuckles' },
|
|
{ slug: 'thumb', name: 'thumb' },
|
|
{ slug: 'index-finger', name: 'index finger' },
|
|
{ slug: 'middle-finger', name: 'middle finger' },
|
|
{ slug: 'ring-finger', name: 'ring finger' },
|
|
{ slug: 'pinky', name: 'pinky' },
|
|
{ slug: 'back-of-hand', name: 'back of hand' },
|
|
{ slug: 'inner-wrist', name: 'inner wrist' },
|
|
{ slug: 'outer-wrist', name: 'outer wrist' },
|
|
// torso
|
|
{ slug: 'shoulder', name: 'shoulder' },
|
|
{ slug: 'collarbone', name: 'collarbone' },
|
|
{ slug: 'chest', name: 'chest' },
|
|
{ slug: 'rib-cage', name: 'rib cage' },
|
|
{ slug: 'breastbone', name: 'breastbone' },
|
|
{ slug: 'underboob', name: 'underboob' },
|
|
{ slug: 'sideboob', name: 'sideboob' },
|
|
{ slug: 'boob', name: 'boob' },
|
|
{ slug: 'nipple', name: 'nipple' },
|
|
{ slug: 'abdomen', name: 'abdomen' },
|
|
{ slug: 'navel', name: 'navel' },
|
|
{ slug: 'pelvis', name: 'pelvis' },
|
|
// back
|
|
{ slug: 'back', name: 'back' },
|
|
{ slug: 'upper-back', name: 'upper back' },
|
|
{ slug: 'middle-back', name: 'lower back' },
|
|
{ slug: 'lower-back', name: 'lower back' },
|
|
{ slug: 'spine', name: 'spine' },
|
|
// bottom
|
|
{ slug: 'butt', name: 'butt' },
|
|
{ slug: 'hip', name: 'hip' },
|
|
{ slug: 'anus', name: 'anus' },
|
|
// genitals
|
|
{ slug: 'pubic-mound', name: 'pubic mound' },
|
|
{ slug: 'vagina', name: 'vagina' },
|
|
{ slug: 'outer-labia', name: 'outer labia' },
|
|
{ slug: 'inner-labia', name: 'inner labia' },
|
|
{ slug: 'clitoris', name: 'clitoris' },
|
|
{ slug: 'penis', name: 'penis' },
|
|
{ slug: 'glans', name: 'glans' },
|
|
{ slug: 'foreskin', name: 'foreskin' },
|
|
{ slug: 'shaft', name: 'shaft' },
|
|
{ slug: 'scrotum', name: 'scrotum' },
|
|
// legs
|
|
{ slug: 'leg', name: 'leg' },
|
|
{ slug: 'groin', name: 'groin' },
|
|
{ slug: 'upper-leg', name: 'upper leg' },
|
|
{ slug: 'thigh', name: 'thigh' },
|
|
{ slug: 'lower-leg', name: 'lower leg' },
|
|
{ slug: 'shin', name: 'shin' },
|
|
{ slug: 'calf', name: 'calf' },
|
|
{ slug: 'knee', name: 'knee' },
|
|
{ slug: 'inner-knee', name: 'inner knee' },
|
|
// feet
|
|
{ slug: 'inner-ankle', name: 'inner ankle' },
|
|
{ slug: 'outer-ankle', name: 'outer ankle' },
|
|
{ slug: 'foot', name: 'foot' },
|
|
{ slug: 'toes', name: 'toes' },
|
|
{ slug: 'big-toe', name: 'big toe' },
|
|
{ slug: 'index-toe', name: 'index toe' },
|
|
{ slug: 'middle-toe', name: 'middle toe' },
|
|
{ slug: 'fourth-toe', name: 'fourth toe' },
|
|
{ slug: 'little-toe', name: 'little toe' },
|
|
]);
|
|
})
|
|
.then(() => knex.schema.createTable('actors_tattoos', (table) => {
|
|
table.increments('id');
|
|
|
|
table.integer('actor_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('actors')
|
|
.onDelete('cascade');
|
|
|
|
table.text('body_slug', 20)
|
|
.references('slug')
|
|
.inTable('body');
|
|
|
|
table.enum('side', ['left', 'right', 'center', 'both']);
|
|
|
|
table.text('description');
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('actors_piercings', (table) => {
|
|
table.increments('id');
|
|
|
|
table.integer('actor_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('actors')
|
|
.onDelete('cascade');
|
|
|
|
table.text('body_slug', 20)
|
|
.references('slug')
|
|
.inTable('body');
|
|
|
|
table.enum('side', ['left', 'right', 'center', 'both']);
|
|
|
|
table.text('description');
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('actors_avatars', (table) => {
|
|
table.integer('profile_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('actors_profiles')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique('profile_id');
|
|
}))
|
|
.then(() => knex.schema.createTable('actors_photos', (table) => {
|
|
table.integer('actor_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('actors')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique(['actor_id', 'media_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('actors_social', (table) => {
|
|
table.increments('id', 16);
|
|
|
|
table.text('url');
|
|
table.text('platform');
|
|
|
|
table.integer('actor_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('actors');
|
|
|
|
table.unique(['url', 'actor_id']);
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('directors', (table) => {
|
|
table.increments('id', 12);
|
|
|
|
table.text('name');
|
|
table.integer('alias_for', 12)
|
|
.references('id')
|
|
.inTable('directors');
|
|
|
|
table.text('slug', 32)
|
|
.unique();
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('releases', (table) => {
|
|
table.increments('id', 16);
|
|
|
|
table.integer('entity_id', 12)
|
|
.references('id')
|
|
.inTable('entities')
|
|
.notNullable();
|
|
|
|
table.integer('studio_id', 12)
|
|
.references('id')
|
|
.inTable('entities');
|
|
|
|
table.text('shoot_id');
|
|
table.text('entry_id');
|
|
|
|
table.unique(['entity_id', 'entry_id']);
|
|
|
|
table.text('url', 1000);
|
|
table.text('title');
|
|
table.text('slug');
|
|
|
|
table.timestamp('date');
|
|
table.index('date');
|
|
|
|
table.date('production_date');
|
|
|
|
table.text('production_location');
|
|
table.text('production_city');
|
|
table.text('production_state');
|
|
table.text('production_country_alpha2', 2)
|
|
.references('alpha2')
|
|
.inTable('countries');
|
|
|
|
table.enum('date_precision', ['year', 'month', 'day', 'hour', 'minute', 'second'])
|
|
.defaultTo('day');
|
|
|
|
table.text('description');
|
|
|
|
table.integer('duration')
|
|
.unsigned();
|
|
|
|
table.boolean('deep');
|
|
table.text('deep_url', 1000);
|
|
|
|
table.text('comment');
|
|
|
|
table.integer('created_batch_id', 12)
|
|
.references('id')
|
|
.inTable('batches')
|
|
.onDelete('cascade');
|
|
|
|
table.integer('updated_batch_id', 12)
|
|
.references('id')
|
|
.inTable('batches')
|
|
.onDelete('cascade');
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_actors', (table) => {
|
|
table.integer('release_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.integer('actor_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('actors')
|
|
.onDelete('cascade');
|
|
|
|
table.unique(['release_id', 'actor_id']);
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_directors', (table) => {
|
|
table.integer('release_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.integer('director_id', 8)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('directors');
|
|
|
|
table.unique(['release_id', 'director_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_posters', (table) => {
|
|
table.integer('release_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique('release_id');
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_covers', (table) => {
|
|
table.integer('release_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique(['release_id', 'media_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_trailers', (table) => {
|
|
table.integer('release_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique('release_id');
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_teasers', (table) => {
|
|
table.integer('release_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique('release_id');
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_photos', (table) => {
|
|
table.integer('release_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique(['release_id', 'media_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_tags', (table) => {
|
|
table.integer('tag_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('tags');
|
|
|
|
table.integer('release_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.unique(['tag_id', 'release_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('releases_search', (table) => {
|
|
table.integer('release_id', 16)
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
}))
|
|
.then(() => knex.schema.createTable('movies', (table) => {
|
|
table.increments('id', 16);
|
|
|
|
table.integer('entity_id', 12)
|
|
.references('id')
|
|
.inTable('entities')
|
|
.notNullable();
|
|
|
|
table.integer('studio_id', 12)
|
|
.references('id')
|
|
.inTable('entities');
|
|
|
|
table.text('entry_id');
|
|
table.unique(['entity_id', 'entry_id']);
|
|
|
|
table.text('url', 1000);
|
|
table.text('title');
|
|
table.text('slug');
|
|
|
|
table.timestamp('date');
|
|
table.index('date');
|
|
|
|
table.enum('date_precision', ['year', 'month', 'day', 'hour', 'minute', 'second'])
|
|
.defaultTo('day');
|
|
|
|
table.text('description');
|
|
|
|
table.boolean('deep');
|
|
table.text('deep_url', 1000);
|
|
|
|
table.text('comment');
|
|
|
|
table.integer('created_batch_id', 12)
|
|
.references('id')
|
|
.inTable('batches')
|
|
.onDelete('cascade');
|
|
|
|
table.integer('updated_batch_id', 12)
|
|
.references('id')
|
|
.inTable('batches')
|
|
.onDelete('cascade');
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('movies_scenes', (table) => {
|
|
table.integer('movie_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('movies')
|
|
.onDelete('cascade');
|
|
|
|
table.integer('scene_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('releases')
|
|
.onDelete('cascade');
|
|
|
|
table.unique(['movie_id', 'scene_id']);
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('movies_covers', (table) => {
|
|
table.integer('movie_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('movies')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique(['movie_id', 'media_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('movies_trailers', (table) => {
|
|
table.integer('movie_id', 16)
|
|
.unique()
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('movies')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
}))
|
|
.then(() => knex.schema.createTable('movies_posters', (table) => {
|
|
table.integer('movie_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('movies')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique(['movie_id', 'media_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('clips', (table) => {
|
|
table.increments('id', 16);
|
|
|
|
table.integer('release_id', 12)
|
|
.references('id')
|
|
.inTable('releases')
|
|
.notNullable()
|
|
.onDelete('cascade');
|
|
|
|
table.integer('clip', 6);
|
|
|
|
table.unique(['release_id', 'clip']);
|
|
|
|
table.text('title');
|
|
table.text('description');
|
|
|
|
table.integer('duration')
|
|
.unsigned();
|
|
|
|
table.integer('created_batch_id', 12)
|
|
.references('id')
|
|
.inTable('batches')
|
|
.onDelete('cascade');
|
|
|
|
table.integer('updated_batch_id', 12)
|
|
.references('id')
|
|
.inTable('batches')
|
|
.onDelete('cascade');
|
|
|
|
table.datetime('created_at')
|
|
.defaultTo(knex.fn.now());
|
|
}))
|
|
.then(() => knex.schema.createTable('clips_posters', (table) => {
|
|
table.integer('clip_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('clips')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique('clip_id');
|
|
}))
|
|
.then(() => knex.schema.createTable('clips_photos', (table) => {
|
|
table.integer('clip_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('clips')
|
|
.onDelete('cascade');
|
|
|
|
table.text('media_id', 21)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('media');
|
|
|
|
table.unique(['clip_id', 'media_id']);
|
|
}))
|
|
.then(() => knex.schema.createTable('clips_tags', (table) => {
|
|
table.integer('tag_id', 12)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('tags')
|
|
.onDelete('cascade');
|
|
|
|
table.integer('clip_id', 16)
|
|
.notNullable()
|
|
.references('id')
|
|
.inTable('clips');
|
|
|
|
table.unique(['tag_id', 'clip_id']);
|
|
}))
|
|
// SEARCH
|
|
.then(() => { // eslint-disable-line arrow-body-style
|
|
// allow vim fold
|
|
return knex.raw(`
|
|
CREATE TEXT SEARCH DICTIONARY traxxx_dict (
|
|
TEMPLATE = pg_catalog.simple,
|
|
stopwords = traxxx
|
|
);
|
|
|
|
CREATE TEXT SEARCH CONFIGURATION traxxx (
|
|
COPY = english
|
|
);
|
|
|
|
ALTER TABLE releases_search
|
|
ADD COLUMN document tsvector;
|
|
|
|
ALTER TEXT SEARCH CONFIGURATION traxxx
|
|
ALTER MAPPING FOR word, numword, hword, numhword, hword_part, hword_numpart, asciiword, asciihword, hword_asciipart WITH traxxx_dict, simple, english_stem;
|
|
`);
|
|
})
|
|
// INDEXES
|
|
.then(() => { // eslint-disable-line arrow-body-style
|
|
// allow vim fold
|
|
return knex.raw(`
|
|
CREATE UNIQUE INDEX unique_actor_slugs_network ON actors (slug, entity_id, entry_id);
|
|
CREATE UNIQUE INDEX unique_actor_slugs ON actors (slug, (entity_id IS NULL));
|
|
|
|
CREATE UNIQUE INDEX releases_search_unique ON releases_search (release_id);
|
|
CREATE INDEX releases_search_index ON releases_search USING GIN (document);
|
|
`);
|
|
})
|
|
// FUNCTIONS
|
|
.then(() => { // eslint-disable-line arrow-body-style
|
|
// allow vim fold
|
|
return knex.raw(`
|
|
CREATE FUNCTION search_releases(query text) RETURNS SETOF releases AS $$
|
|
SELECT * FROM releases WHERE releases.id IN (
|
|
SELECT release_id FROM releases_search AS search
|
|
WHERE search.document @@ plainto_tsquery('traxxx', regexp_replace(query, '\\.|-|(XXX\\.[\\d+|hd|sd].*$)', ' ', 'ig'))
|
|
ORDER BY ts_rank(search.document, plainto_tsquery('traxxx', regexp_replace(query, '\\.|-|(XXX\\.[\\d+|hd|sd].*$)', ' ', 'ig'))) DESC
|
|
);
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION search_entities(search text) RETURNS SETOF entities AS $$
|
|
SELECT * FROM entities
|
|
WHERE
|
|
name ILIKE ('%' || search || '%') OR
|
|
slug ILIKE ('%' || search || '%') OR
|
|
array_to_string(alias, '') ILIKE ('%' || search || '%') OR
|
|
replace(array_to_string(alias, ''), ' ', '') ILIKE ('%' || search || '%') OR
|
|
url ILIKE ('%' || search || '%')
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION search_actors(search text, min_length numeric DEFAULT 2) RETURNS SETOF actors AS $$
|
|
SELECT * FROM actors
|
|
WHERE length(search) >= min_length
|
|
AND name ILIKE ('%' || search || '%')
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION actors_tags(actor actors, selectable_tags text[]) RETURNS SETOF tags AS $$
|
|
SELECT tags.*
|
|
FROM releases_actors
|
|
LEFT JOIN
|
|
releases_tags ON releases_tags.release_id = releases_actors.release_id
|
|
LEFT JOIN
|
|
tags ON tags.id = releases_tags.tag_id
|
|
WHERE
|
|
releases_actors.actor_id = actor.id
|
|
AND
|
|
CASE WHEN array_length(selectable_tags, 1) IS NOT NULL
|
|
THEN tags.slug = ANY(selectable_tags)
|
|
ELSE true
|
|
END
|
|
GROUP BY tags.id
|
|
ORDER BY tags.name;
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION actors_channels(actor actors) RETURNS SETOF entities AS $$
|
|
SELECT entities.*
|
|
FROM releases_actors
|
|
LEFT JOIN releases ON releases.id = releases_actors.release_id
|
|
LEFT JOIN entities ON entities.id = releases.entity_id
|
|
WHERE releases_actors.actor_id = actor.id
|
|
GROUP BY entities.id;
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION actors_actors(actor actors) RETURNS SETOF actors AS $$
|
|
SELECT actors.*
|
|
FROM releases_actors
|
|
LEFT JOIN releases_actors AS associated_actors ON associated_actors.release_id = releases_actors.release_id
|
|
LEFT JOIN actors ON actors.id = associated_actors.actor_id
|
|
WHERE releases_actors.actor_id = actor.id
|
|
AND NOT actors.id = actor.id
|
|
GROUP BY actors.id
|
|
ORDER BY actors.name;
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
/* GraphQL/Postgraphile 'every' applies to the data, will only include scenes for which every assigned tag is selected,
|
|
instead of what we want; scenes with every selected tag, but possibly also some others */
|
|
CREATE FUNCTION actors_scenes(actor actors, selected_tags text[], mode text DEFAULT 'all') RETURNS SETOF releases AS $$
|
|
SELECT releases.*
|
|
FROM releases
|
|
LEFT JOIN
|
|
releases_actors ON releases_actors.release_id = releases.id
|
|
LEFT JOIN
|
|
releases_tags ON releases_tags.release_id = releases.id
|
|
LEFT JOIN
|
|
tags ON tags.id = releases_tags.tag_id
|
|
WHERE releases_actors.actor_id = actor.id
|
|
AND CASE
|
|
/* match at least one of the selected tags */
|
|
WHEN mode = 'any'
|
|
AND array_length(selected_tags, 1) > 0
|
|
THEN tags.slug = ANY(selected_tags)
|
|
ELSE true
|
|
END
|
|
GROUP BY releases.id
|
|
HAVING CASE
|
|
/* match all of the selected tags */
|
|
WHEN mode = 'all'
|
|
AND array_length(selected_tags, 1) > 0
|
|
THEN COUNT(
|
|
CASE WHEN tags.slug = ANY(selected_tags)
|
|
THEN true
|
|
END
|
|
) = array_length(selected_tags, 1)
|
|
ELSE true
|
|
END;
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION tags_scenes(tag tags, selected_tags text[], mode text DEFAULT 'all') RETURNS SETOF releases AS $$
|
|
SELECT releases.*
|
|
FROM releases
|
|
LEFT JOIN
|
|
releases_actors ON releases_actors.release_id = releases.id
|
|
LEFT JOIN
|
|
releases_tags ON releases_tags.release_id = releases.id
|
|
LEFT JOIN
|
|
tags ON tags.id = releases_tags.tag_id
|
|
WHERE releases_tags.tag_id = tag.id
|
|
GROUP BY releases.id;
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION movies_actors(movie movies) RETURNS SETOF actors AS $$
|
|
SELECT actors.*
|
|
FROM movies_scenes
|
|
LEFT JOIN
|
|
releases ON releases.id = movies_scenes.scene_id
|
|
LEFT JOIN
|
|
releases_actors ON releases_actors.release_id = releases.id
|
|
LEFT JOIN
|
|
actors ON actors.id = releases_actors.actor_id
|
|
WHERE movies_scenes.movie_id = movie.id
|
|
AND actors.id IS NOT NULL
|
|
GROUP BY actors.id
|
|
ORDER BY actors.name, actors.gender
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION movies_tags(movie movies) RETURNS SETOF tags AS $$
|
|
SELECT tags.*
|
|
FROM movies_scenes
|
|
LEFT JOIN
|
|
releases ON releases.id = movies_scenes.scene_id
|
|
LEFT JOIN
|
|
releases_tags ON releases_tags.release_id = releases.id
|
|
LEFT JOIN
|
|
tags ON tags.id = releases_tags.tag_id
|
|
WHERE movies_scenes.movie_id = movie.id
|
|
AND tags.id IS NOT NULL
|
|
GROUP BY tags.id
|
|
ORDER BY tags.priority DESC
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION movies_photos(movie movies) RETURNS SETOF media AS $$
|
|
SELECT media.*
|
|
FROM movies_scenes
|
|
LEFT JOIN
|
|
releases ON releases.id = movies_scenes.scene_id
|
|
INNER JOIN
|
|
releases_photos ON releases_photos.release_id = releases.id
|
|
LEFT JOIN
|
|
media ON media.id = releases_photos.media_id
|
|
WHERE movies_scenes.movie_id = movie.id
|
|
GROUP BY media.id
|
|
ORDER BY media.index ASC
|
|
$$ LANGUAGE SQL STABLE;
|
|
|
|
CREATE FUNCTION releases_is_new(release releases) RETURNS boolean AS $$
|
|
SELECT EXISTS(SELECT true WHERE (SELECT id FROM batches ORDER BY created_at DESC LIMIT 1) = release.created_batch_id);
|
|
$$ LANGUAGE sql STABLE;
|
|
`);
|
|
})
|
|
// VIEWS AND COMMENTS
|
|
.then(() => { // eslint-disable-line arrow-body-style
|
|
// allow vim fold
|
|
return knex.raw(`
|
|
COMMENT ON COLUMN actors.height IS E'@omit read,update,create,delete,all,many';
|
|
COMMENT ON COLUMN actors.weight IS E'@omit read,update,create,delete,all,many';
|
|
COMMENT ON COLUMN actors.penis_length IS E'@omit read,update,create,delete,all,many';
|
|
COMMENT ON COLUMN actors.penis_girth IS E'@omit read,update,create,delete,all,many';
|
|
|
|
COMMENT ON FUNCTION actors_tags IS E'@sortable';
|
|
COMMENT ON FUNCTION actors_channels IS E'@sortable';
|
|
COMMENT ON FUNCTION actors_actors IS E'@sortable';
|
|
COMMENT ON FUNCTION actors_scenes IS E'@sortable';
|
|
|
|
COMMENT ON FUNCTION tags_scenes IS E'@sortable';
|
|
`);
|
|
});
|
|
|
|
exports.down = (knex) => { // eslint-disable-line arrow-body-style
|
|
// allow vim fold
|
|
return knex.raw(`
|
|
DROP TABLE IF EXISTS releases_actors CASCADE;
|
|
DROP TABLE IF EXISTS releases_movies CASCADE;
|
|
DROP TABLE IF EXISTS releases_directors CASCADE;
|
|
DROP TABLE IF EXISTS releases_posters CASCADE;
|
|
DROP TABLE IF EXISTS releases_photos CASCADE;
|
|
DROP TABLE IF EXISTS releases_covers CASCADE;
|
|
DROP TABLE IF EXISTS releases_trailers CASCADE;
|
|
DROP TABLE IF EXISTS releases_teasers CASCADE;
|
|
DROP TABLE IF EXISTS releases_tags CASCADE;
|
|
DROP TABLE IF EXISTS releases_search CASCADE;
|
|
|
|
DROP TABLE IF EXISTS movies_scenes CASCADE;
|
|
DROP TABLE IF EXISTS movies_covers CASCADE;
|
|
DROP TABLE IF EXISTS movies_posters CASCADE;
|
|
DROP TABLE IF EXISTS movies_trailers CASCADE;
|
|
|
|
DROP TABLE IF EXISTS clips_tags CASCADE;
|
|
DROP TABLE IF EXISTS clips_posters CASCADE;
|
|
DROP TABLE IF EXISTS clips_photos CASCADE;
|
|
|
|
DROP TABLE IF EXISTS batches CASCADE;
|
|
|
|
DROP TABLE IF EXISTS actors_avatars CASCADE;
|
|
DROP TABLE IF EXISTS actors_photos CASCADE;
|
|
DROP TABLE IF EXISTS actors_social CASCADE;
|
|
DROP TABLE IF EXISTS actors_profiles CASCADE;
|
|
DROP TABLE IF EXISTS actors_tattoos CASCADE;
|
|
DROP TABLE IF EXISTS actors_piercings CASCADE;
|
|
DROP TABLE IF EXISTS body CASCADE;
|
|
|
|
DROP TABLE IF EXISTS entities_tags CASCADE;
|
|
DROP TABLE IF EXISTS entities_social CASCADE;
|
|
DROP TABLE IF EXISTS sites_tags CASCADE;
|
|
DROP TABLE IF EXISTS sites_social CASCADE;
|
|
DROP TABLE IF EXISTS networks_social CASCADE;
|
|
DROP TABLE IF EXISTS tags_posters CASCADE;
|
|
DROP TABLE IF EXISTS tags_photos CASCADE;
|
|
DROP TABLE IF EXISTS movies CASCADE;
|
|
DROP TABLE IF EXISTS clips CASCADE;
|
|
DROP TABLE IF EXISTS releases CASCADE;
|
|
DROP TABLE IF EXISTS actors CASCADE;
|
|
DROP TABLE IF EXISTS directors CASCADE;
|
|
DROP TABLE IF EXISTS tags CASCADE;
|
|
DROP TABLE IF EXISTS tags_groups CASCADE;
|
|
DROP TABLE IF EXISTS social CASCADE;
|
|
DROP TABLE IF EXISTS sites CASCADE;
|
|
DROP TABLE IF EXISTS studios CASCADE;
|
|
DROP TABLE IF EXISTS media_sfw CASCADE;
|
|
DROP TABLE IF EXISTS media CASCADE;
|
|
DROP TABLE IF EXISTS countries CASCADE;
|
|
DROP TABLE IF EXISTS networks CASCADE;
|
|
|
|
DROP TABLE IF EXISTS entities_types CASCADE;
|
|
DROP TABLE IF EXISTS entities CASCADE;
|
|
|
|
DROP FUNCTION IF EXISTS search_sites;
|
|
DROP FUNCTION IF EXISTS search_entities;
|
|
DROP FUNCTION IF EXISTS search_actors;
|
|
DROP FUNCTION IF EXISTS get_random_sfw_media_id;
|
|
|
|
DROP FUNCTION IF EXISTS releases_is_new;
|
|
DROP FUNCTION IF EXISTS actors_tags;
|
|
DROP FUNCTION IF EXISTS actors_channels;
|
|
DROP FUNCTION IF EXISTS actors_actors;
|
|
DROP FUNCTION IF EXISTS actors_scenes;
|
|
|
|
DROP FUNCTION IF EXISTS movies_actors;
|
|
DROP FUNCTION IF EXISTS movies_tags;
|
|
DROP FUNCTION IF EXISTS movies_photos;
|
|
|
|
DROP TEXT SEARCH CONFIGURATION IF EXISTS traxxx;
|
|
DROP TEXT SEARCH DICTIONARY IF EXISTS traxxx_dict;
|
|
`);
|
|
};
|