Added rudimentary affiliate banner setup. Separated login and signup disable. Added various tag photos.
|  | @ -36,6 +36,7 @@ | |||
| 			>Log in</button> | ||||
| 
 | ||||
| 			<router-link | ||||
| 				v-if="$store.state.auth.signup" | ||||
| 				:to="{ name: 'signup', query: { ref: $route.query.ref } }" | ||||
| 				class="link link-external signup" | ||||
| 			>Sign up</router-link> | ||||
|  | @ -65,7 +66,7 @@ async function login() { | |||
| } | ||||
| 
 | ||||
| function mounted() { | ||||
| 	if (!this.$store.state.auth.enabled) { | ||||
| 	if (!this.$store.state.auth.login) { | ||||
| 		this.$router.replace({ name: 'not-found' }); | ||||
| 	} | ||||
| } | ||||
|  |  | |||
|  | @ -44,6 +44,7 @@ | |||
| 			>Sign up</button> | ||||
| 
 | ||||
| 			<router-link | ||||
| 				v-if="$store.state.auth.login" | ||||
| 				:to="{ name: 'login', query: { ref: $route.query.ref } }" | ||||
| 				class="link link-external login" | ||||
| 			>Log in</router-link> | ||||
|  | @ -74,7 +75,7 @@ async function signup() { | |||
| } | ||||
| 
 | ||||
| function mounted() { | ||||
| 	if (!this.$store.state.auth.enabled) { | ||||
| 	if (!this.$store.state.auth.signup) { | ||||
| 		this.$router.replace({ name: 'not-found' }); | ||||
| 	} | ||||
| } | ||||
|  |  | |||
|  | @ -5,7 +5,7 @@ | |||
| 	> | ||||
| 		<div class="info"> | ||||
| 			<a | ||||
| 				:href="entity.url" | ||||
| 				:href="entityUrl" | ||||
| 				target="_blank" | ||||
| 				rel="noopener" | ||||
| 				class="link link-child" | ||||
|  | @ -41,6 +41,25 @@ | |||
| 				/> | ||||
| 			</a> | ||||
| 
 | ||||
| 			<a | ||||
| 				v-if="campaign" | ||||
| 				:href="campaign.affiliate.url" | ||||
| 				target="_blank" | ||||
| 				class="campaign" | ||||
| 			> | ||||
| 				<img | ||||
| 					v-if="campaign.banner.entity.type === 'network'" | ||||
| 					:src="`/img/banners/${campaign.banner.entity.slug}/${campaign.banner.id}.jpeg`" | ||||
| 					class="campaign-banner" | ||||
| 				> | ||||
| 
 | ||||
| 				<img | ||||
| 					v-if="campaign.banner.entity.type === 'channel' && campaign.banner.entity.parent?.type === 'network'" | ||||
| 					:src="`/img/banners/${campaign.banner.entity.parent.slug}/${campaign.banner.entity.slug}/${campaign.banner.id}.jpeg`" | ||||
| 					class="campaign-banner" | ||||
| 				> | ||||
| 			</a> | ||||
| 
 | ||||
| 			<ul | ||||
| 				v-if="entity.tags.length > 0" | ||||
| 				class="tags" | ||||
|  | @ -72,6 +91,8 @@ | |||
| 					v-else | ||||
| 					class="name parent-name" | ||||
| 				>{{ entity.parent.name }}</h3> | ||||
| 
 | ||||
| 				<Icon icon="device_hub" /> | ||||
| 			</router-link> | ||||
| 		</div> | ||||
| 
 | ||||
|  | @ -124,6 +145,9 @@ import Children from './children.vue'; | |||
| import Scroll from '../scroll/scroll.vue'; | ||||
| 
 | ||||
| async function fetchEntity(scroll = true) { | ||||
| 	this.campaign = null; | ||||
| 	this.entityUrl = null; | ||||
| 
 | ||||
| 	const { entity, totalCount } = await this.$store.dispatch('fetchEntityBySlugAndType', { | ||||
| 		entitySlug: this.$route.params.entitySlug, | ||||
| 		entityType: this.$route.name, | ||||
|  | @ -137,6 +161,16 @@ async function fetchEntity(scroll = true) { | |||
| 
 | ||||
| 	this.pageTitle = entity.name; | ||||
| 
 | ||||
| 	const channelBannerCampaigns = entity.campaigns.filter(campaign => campaign.banner); | ||||
| 	const networkBannerCampaigns = entity.parent?.campaigns.filter(campaign => campaign.banner); | ||||
| 	const bannerCampaigns = channelBannerCampaigns.length > 0 ? channelBannerCampaigns : networkBannerCampaigns; | ||||
| 
 | ||||
| 	if (bannerCampaigns.length > 0) { | ||||
| 		this.campaign = bannerCampaigns[Math.floor(Math.random() * bannerCampaigns.length)]; | ||||
| 	} | ||||
| 
 | ||||
| 	this.entityUrl = entity.campaigns.find(campaign => !campaign.banner)?.affiliate.url || entity.url; | ||||
| 
 | ||||
| 	if (scroll && this.$refs.filter?.$el) { | ||||
| 		this.$refs.filter.$el.scrollIntoView(); | ||||
| 	} | ||||
|  | @ -168,6 +202,8 @@ export default { | |||
| 			totalCount: null, | ||||
| 			limit: Number(this.$route.query.limit) || 20, | ||||
| 			expanded: false, | ||||
| 			campaign: null, | ||||
| 			entityUrl: null, | ||||
| 		}; | ||||
| 	}, | ||||
| 	watch: { | ||||
|  | @ -185,6 +221,7 @@ export default { | |||
| @import 'breakpoints'; | ||||
| 
 | ||||
| .info { | ||||
| 	height: 6rem; | ||||
| 	display: flex; | ||||
| 	justify-content: space-between; | ||||
| 	background: var(--profile); | ||||
|  | @ -214,6 +251,13 @@ export default { | |||
| .link-parent { | ||||
| 	flex-direction: row-reverse; | ||||
| 	margin: 0 0 0 3rem; | ||||
| 
 | ||||
| 	.icon { | ||||
| 		width: 1.25rem; | ||||
| 		height: 1.25rem; | ||||
| 		fill: var(--lighten); | ||||
| 		margin: 0 .5rem 0 0; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| .logo { | ||||
|  | @ -223,7 +267,7 @@ export default { | |||
| } | ||||
| 
 | ||||
| .logo-child { | ||||
| 	height: 2rem; | ||||
| 	height: 3rem; | ||||
| } | ||||
| 
 | ||||
| .logo-parent { | ||||
|  | @ -234,6 +278,14 @@ export default { | |||
| 	height: 1rem; | ||||
| } | ||||
| 
 | ||||
| .campaign { | ||||
| 	height: 100%; | ||||
| } | ||||
| 
 | ||||
| .campaign-banner { | ||||
| 	height: 100%; | ||||
| } | ||||
| 
 | ||||
| .content-inner { | ||||
| 	display: flex; | ||||
| 	flex-direction: column; | ||||
|  |  | |||
|  | @ -2,13 +2,13 @@ | |||
| 	<div class="menu"> | ||||
| 		<ul class="menu-items noselect"> | ||||
| 			<router-link | ||||
| 				v-if="auth && me" | ||||
| 				v-if="login && me" | ||||
| 				:to="{ name: 'user', params: { username: me.username } }" | ||||
| 				class="menu-username" | ||||
| 			>{{ me.username }}</router-link> | ||||
| 
 | ||||
| 			<router-link | ||||
| 				v-else-if="auth" | ||||
| 				v-else-if="login" | ||||
| 				:to="{ name: 'login', query: { ref: $route.path } }" | ||||
| 				class="menu-item" | ||||
| 				@click.stop | ||||
|  | @ -17,7 +17,7 @@ | |||
| 			</router-link> | ||||
| 
 | ||||
| 			<li | ||||
| 				v-if="auth && me" | ||||
| 				v-if="login && me" | ||||
| 				class="menu-item" | ||||
| 				@click.stop="$store.dispatch('logout')" | ||||
| 			> | ||||
|  | @ -89,8 +89,12 @@ function theme(state) { | |||
| 	return state.ui.theme; | ||||
| } | ||||
| 
 | ||||
| function auth(state) { | ||||
| 	return state.auth.enabled; | ||||
| function login(state) { | ||||
| 	return state.auth.login; | ||||
| } | ||||
| 
 | ||||
| function signup(state) { | ||||
| 	return state.auth.signup; | ||||
| } | ||||
| 
 | ||||
| function me(state) { | ||||
|  | @ -108,7 +112,8 @@ function setSfw(enabled) { | |||
| export default { | ||||
| 	computed: { | ||||
| 		...mapState({ | ||||
| 			auth, | ||||
| 			login, | ||||
| 			signup, | ||||
| 			sfw, | ||||
| 			theme, | ||||
| 			me, | ||||
|  |  | |||
|  | @ -47,6 +47,27 @@ | |||
| 				@close="$router.replace({ hash: undefined })" | ||||
| 			/> | ||||
| 
 | ||||
| 			<div class="campaign-container"> | ||||
| 				<a | ||||
| 					v-if="campaign" | ||||
| 					:href="campaign.affiliate.url" | ||||
| 					target="_blank" | ||||
| 					class="campaign" | ||||
| 				> | ||||
| 					<img | ||||
| 						v-if="campaign.banner.entity.type === 'network'" | ||||
| 						:src="`/img/banners/${campaign.banner.entity.slug}/${campaign.banner.id}.jpeg`" | ||||
| 						class="campaign-banner" | ||||
| 					> | ||||
| 
 | ||||
| 					<img | ||||
| 						v-if="campaign.banner.entity.type === 'channel' && campaign.banner.entity.parent?.type === 'network'" | ||||
| 						:src="`/img/banners/${campaign.banner.entity.parent.slug}/${campaign.banner.entity.slug}/${campaign.banner.id}.jpeg`" | ||||
| 						class="campaign-banner" | ||||
| 					> | ||||
| 				</a> | ||||
| 			</div> | ||||
| 
 | ||||
| 			<FilterBar | ||||
| 				ref="filter" | ||||
| 				:fetch-releases="fetchReleases" | ||||
|  | @ -94,6 +115,15 @@ async function fetchReleases(scroll = true) { | |||
| 	this.hasMedia = this.tag.poster || this.tag.photos.length > 0; | ||||
| 	this.description = this.tag.description && converter.makeHtml(escapeHtml(this.tag.description)); | ||||
| 
 | ||||
| 	if (tag.banners.length > 0) { | ||||
| 		const banner = tag.banners[Math.floor(Math.random() * tag.banners.length)]; | ||||
| 
 | ||||
| 		this.campaign = { | ||||
| 			...banner.campaigns[Math.floor(Math.random() * banner.campaigns.length)], | ||||
| 			banner, | ||||
| 		}; | ||||
| 	} | ||||
| 
 | ||||
| 	if (scroll && this.$refs.filter) { | ||||
| 		this.$refs.filter.$el.scrollIntoView(); | ||||
| 	} | ||||
|  | @ -133,6 +163,7 @@ export default { | |||
| 			pageTitle: null, | ||||
| 			hasMedia: false, | ||||
| 			expanded: false, | ||||
| 			campaign: null, | ||||
| 		}; | ||||
| 	}, | ||||
| 	computed: { | ||||
|  | @ -172,12 +203,14 @@ export default { | |||
| @import 'theme'; | ||||
| 
 | ||||
| .header { | ||||
| 	display: flex; | ||||
|     background: var(--profile); | ||||
|     color: var(--text-light); | ||||
|     justify-content: space-between; | ||||
| } | ||||
| 
 | ||||
| .title { | ||||
| 	display: inline-block; | ||||
|     padding: .5rem 1rem; | ||||
|     margin: 0; | ||||
|     flex-shrink: 0; | ||||
|  | @ -200,4 +233,21 @@ export default { | |||
| .scroll { | ||||
| 	background: var(--background-dim); | ||||
| } | ||||
| 
 | ||||
| .campaign-container { | ||||
| 	max-height: 90px; | ||||
| 	padding: .5rem 1rem 0 1rem; | ||||
| 	background: var(--background-dim); | ||||
| 	text-align: center; | ||||
| } | ||||
| 
 | ||||
| .campaign { | ||||
| 	display: inline-block; | ||||
| 	height: 100%; | ||||
| 
 | ||||
| 	.campaign-banner { | ||||
| 		max-height: 100%; | ||||
| 		max-width: 100%; | ||||
| 	} | ||||
| } | ||||
| </style> | ||||
|  |  | |||
|  | @ -1,4 +1,5 @@ | |||
| export default { | ||||
| 	enabled: window.env.auth, | ||||
| 	login: window.env.login, | ||||
| 	signup: window.env.signup, | ||||
| 	user: null, | ||||
| }; | ||||
|  |  | |||
|  | @ -126,6 +126,7 @@ function curateTag(tag) { | |||
| 	}; | ||||
| 
 | ||||
| 	if (tag.releases) curatedTag.releases = tag.releases.map(({ release }) => curateRelease(release)); | ||||
| 	if (tag.banners) curatedTag.banners = tag.banners.map(({ banner }) => banner); | ||||
| 	if (tag.photos) curatedTag.photos = tag.photos.map(({ media }) => media); | ||||
| 	if (tag.poster) curatedTag.poster = tag.poster.media; | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,6 +1,6 @@ | |||
| import { graphql } from '../api'; | ||||
| // import { sitesFragment, releaseFields } from '../fragments';
 | ||||
| import { releaseFields } from '../fragments'; | ||||
| import { releaseFields, campaignsFragment } from '../fragments'; | ||||
| import { curateEntity } from '../curate'; | ||||
| import getDateRange from '../get-date-range'; | ||||
| 
 | ||||
|  | @ -65,6 +65,7 @@ function initEntitiesActions(store, router) { | |||
| 							hasLogo | ||||
| 						} | ||||
|                     } | ||||
| 					${campaignsFragment} | ||||
|                     parent { | ||||
|                         id | ||||
|                         name | ||||
|  | @ -73,6 +74,7 @@ function initEntitiesActions(store, router) { | |||
|                         url | ||||
| 						independent | ||||
| 						hasLogo | ||||
| 						${campaignsFragment} | ||||
|                     } | ||||
|                 } | ||||
| 				connection: releasesConnection( | ||||
|  |  | |||
|  | @ -96,6 +96,56 @@ const actorFields = ` | |||
| 		${actorStashesFields} | ||||
| `;
 | ||||
| 
 | ||||
| const campaignsFragment = ` | ||||
| 	campaigns(filter: { | ||||
| 		banner: { | ||||
| 			bannersTagsConnection: { | ||||
| 				none: { | ||||
| 					tag: { | ||||
| 						slug: { | ||||
| 							in: $exclude | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}) { | ||||
| 		affiliate { | ||||
| 			id | ||||
| 			url | ||||
| 		} | ||||
| 		banner { | ||||
| 			id | ||||
| 			width | ||||
| 			height | ||||
| 			entity { | ||||
| 				id | ||||
| 				type | ||||
| 				name | ||||
| 				slug | ||||
| 				parent { | ||||
| 					id | ||||
| 					type | ||||
| 					name | ||||
| 					slug | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		entity { | ||||
| 			id | ||||
| 			type | ||||
| 			name | ||||
| 			slug | ||||
| 			parent { | ||||
| 				id | ||||
| 				type | ||||
| 				name | ||||
| 				slug | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| `;
 | ||||
| 
 | ||||
| const releaseActorsFragment = ` | ||||
|     actors: releasesActors(orderBy: ACTOR_BY_ACTOR_ID__GENDER_ASC) { | ||||
|       actor { | ||||
|  | @ -480,6 +530,7 @@ function getIncludedActors(router) { | |||
| export { | ||||
| 	actorFields, | ||||
| 	actorStashesFields, | ||||
| 	campaignsFragment, | ||||
| 	releaseActorsFragment, | ||||
| 	releaseFields, | ||||
| 	releaseTagsFragment, | ||||
|  |  | |||
|  | @ -103,6 +103,57 @@ function initTagsActions(store, _router) { | |||
|                             } | ||||
|                         } | ||||
|                     } | ||||
| 					banners: bannersTags(filter: { | ||||
| 						banner: { | ||||
| 							bannersTagsConnection: { | ||||
| 								none: { | ||||
| 									tag: { | ||||
| 										slug: { | ||||
| 											in: $exclude | ||||
| 										} | ||||
| 									} | ||||
| 								} | ||||
| 							} | ||||
| 						} | ||||
| 					}) { | ||||
| 						banner { | ||||
| 							id | ||||
| 							width | ||||
| 							height | ||||
| 							entity { | ||||
| 								id | ||||
| 								type | ||||
| 								name | ||||
| 								slug | ||||
| 								independent | ||||
| 								parent { | ||||
| 									id | ||||
| 									type | ||||
| 									name | ||||
| 									slug | ||||
| 									independent | ||||
| 								} | ||||
| 							} | ||||
| 							campaigns { | ||||
| 								affiliate { | ||||
| 									id | ||||
| 									url | ||||
| 								} | ||||
| 								entity { | ||||
| 									id | ||||
| 									type | ||||
| 									name | ||||
| 									slug | ||||
| 									parent { | ||||
| 										id | ||||
| 										type | ||||
| 										name | ||||
| 										slug | ||||
| 									} | ||||
| 								} | ||||
| 							} | ||||
| 						} | ||||
| 					} | ||||
|                     scenesConnection( | ||||
|                         filter: { | ||||
| 							date: { | ||||
|  |  | |||
|  | @ -35,7 +35,8 @@ module.exports = { | |||
| 		secretKey: 'abcdefghijklmnopqrstuvwxyz1234567890ABCD', | ||||
| 	}, | ||||
| 	auth: { | ||||
| 		enabled: true, | ||||
| 		login: true, | ||||
| 		signup: true, | ||||
| 	}, | ||||
| 	exclude: { | ||||
| 		channels: [ | ||||
|  |  | |||
|  | @ -1274,6 +1274,83 @@ exports.up = knex => Promise.resolve() | |||
| 			.notNullable() | ||||
| 			.defaultTo(knex.fn.now()); | ||||
| 	})) | ||||
| 	.then(() => knex.schema.createTable('affiliates', (table) => { | ||||
| 		table.string('id') | ||||
| 			.primary() | ||||
| 			.unique() | ||||
| 			.notNullable(); | ||||
| 
 | ||||
| 		table.text('url') | ||||
| 			.notNullable(); | ||||
| 
 | ||||
| 		table.text('comment'); | ||||
| 
 | ||||
| 		table.datetime('created_at') | ||||
| 			.notNullable() | ||||
| 			.defaultTo(knex.fn.now()); | ||||
| 	})) | ||||
| 	.then(() => knex.schema.createTable('banners', (table) => { | ||||
| 		table.string('id') | ||||
| 			.primary() | ||||
| 			.unique() | ||||
| 			.notNullable(); | ||||
| 
 | ||||
| 		table.integer('width') | ||||
| 			.notNullable(); | ||||
| 
 | ||||
| 		table.integer('height') | ||||
| 			.notNullable(); | ||||
| 
 | ||||
| 		table.integer('entity_id', 12) | ||||
| 			.references('id') | ||||
| 			.inTable('entities'); | ||||
| 
 | ||||
| 		table.text('comment'); | ||||
| 
 | ||||
| 		table.datetime('created_at') | ||||
| 			.notNullable() | ||||
| 			.defaultTo(knex.fn.now()); | ||||
| 	})) | ||||
| 	.then(() => knex.schema.createTable('banners_tags', (table) => { | ||||
| 		table.increments('id'); | ||||
| 
 | ||||
| 		table.string('banner_id') | ||||
| 			.notNullable() | ||||
| 			.references('id') | ||||
| 			.inTable('banners'); | ||||
| 
 | ||||
| 		table.integer('tag_id') | ||||
| 			.notNullable() | ||||
| 			.references('id') | ||||
| 			.inTable('tags'); | ||||
| 
 | ||||
| 		table.unique(['banner_id', 'tag_id']); | ||||
| 
 | ||||
| 		table.datetime('created_at') | ||||
| 			.notNullable() | ||||
| 			.defaultTo(knex.fn.now()); | ||||
| 	})) | ||||
| 	.then(() => knex.schema.createTable('campaigns', (table) => { | ||||
| 		table.increments('id'); | ||||
| 
 | ||||
| 		table.integer('entity_id', 12) | ||||
| 			.notNullable() | ||||
| 			.references('id') | ||||
| 			.inTable('entities'); | ||||
| 
 | ||||
| 		table.string('affiliate_id') | ||||
| 			.notNullable() | ||||
| 			.references('id') | ||||
| 			.inTable('affiliates'); | ||||
| 
 | ||||
| 		table.string('banner_id') | ||||
| 			.references('id') | ||||
| 			.inTable('banners'); | ||||
| 
 | ||||
| 		table.datetime('created_at') | ||||
| 			.notNullable() | ||||
| 			.defaultTo(knex.fn.now()); | ||||
| 	})) | ||||
| 	// SEARCH AND SORT
 | ||||
| 	.then(() => { // eslint-disable-line arrow-body-style
 | ||||
| 		// allow vim fold
 | ||||
|  | @ -1293,6 +1370,9 @@ exports.up = knex => Promise.resolve() | |||
| 			CREATE UNIQUE INDEX unique_actor_slugs_network ON actors (slug, entity_id, entry_id); | ||||
| 			CREATE UNIQUE INDEX unique_actor_slugs ON actors (slug) WHERE entity_id IS NULL; | ||||
| 
 | ||||
| 			CREATE UNIQUE INDEX unique_entity_campaigns_banner ON campaigns (entity_id, affiliate_id, banner_id); | ||||
| 			CREATE UNIQUE INDEX unique_entity_campaigns ON campaigns (entity_id, affiliate_id) WHERE banner_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); | ||||
| 		`);
 | ||||
|  | @ -1647,6 +1727,10 @@ exports.down = (knex) => { // eslint-disable-line arrow-body-style | |||
| 		DROP TABLE IF EXISTS chapters_posters CASCADE; | ||||
| 		DROP TABLE IF EXISTS chapters_photos CASCADE; | ||||
| 
 | ||||
| 		DROP TABLE IF EXISTS banners_tags CASCADE; | ||||
| 		DROP TABLE IF EXISTS banners CASCADE; | ||||
| 		DROP TABLE IF EXISTS affiliates CASCADE; | ||||
| 		DROP TABLE IF EXISTS campaigns CASCADE; | ||||
| 		DROP TABLE IF EXISTS batches CASCADE; | ||||
| 
 | ||||
| 		DROP TABLE IF EXISTS actors_avatars CASCADE; | ||||
|  | @ -1659,6 +1743,7 @@ exports.down = (knex) => { // eslint-disable-line arrow-body-style | |||
| 
 | ||||
| 		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; | ||||
|  |  | |||
| After Width: | Height: | Size: 87 KiB | 
| After Width: | Height: | Size: 83 KiB | 
| After Width: | Height: | Size: 90 KiB | 
| After Width: | Height: | Size: 70 KiB | 
| Before Width: | Height: | Size: 81 KiB After Width: | Height: | Size: 81 KiB | 
| After Width: | Height: | Size: 72 KiB | 
| After Width: | Height: | Size: 72 KiB | 
| After Width: | Height: | Size: 72 KiB | 
| After Width: | Height: | Size: 9.4 KiB | 
| After Width: | Height: | Size: 2.3 MiB | 
| After Width: | Height: | Size: 42 KiB | 
| After Width: | Height: | Size: 12 MiB | 
| After Width: | Height: | Size: 8.1 KiB | 
| After Width: | Height: | Size: 33 KiB | 
| After Width: | Height: | Size: 444 KiB | 
| After Width: | Height: | Size: 9.1 KiB | 
| After Width: | Height: | Size: 48 KiB | 
| After Width: | Height: | Size: 11 MiB | 
| After Width: | Height: | Size: 9.9 KiB | 
| After Width: | Height: | Size: 46 KiB | 
| After Width: | Height: | Size: 2.0 MiB | 
| After Width: | Height: | Size: 1.4 MiB | 
| After Width: | Height: | Size: 9.4 KiB | 
| After Width: | Height: | Size: 8.9 KiB | 
| After Width: | Height: | Size: 44 KiB | 
| After Width: | Height: | Size: 42 KiB | 
| After Width: | Height: | Size: 7.8 KiB | 
| After Width: | Height: | Size: 3.0 MiB | 
| After Width: | Height: | Size: 36 KiB | 
| After Width: | Height: | Size: 12 MiB | 
| After Width: | Height: | Size: 14 MiB | 
| After Width: | Height: | Size: 13 MiB | 
| After Width: | Height: | Size: 11 MiB | 
| After Width: | Height: | Size: 11 MiB | 
| After Width: | Height: | Size: 10 KiB | 
| After Width: | Height: | Size: 10 KiB | 
| After Width: | Height: | Size: 11 KiB | 
| After Width: | Height: | Size: 9.7 KiB | 
| After Width: | Height: | Size: 9.8 KiB | 
| After Width: | Height: | Size: 49 KiB | 
| After Width: | Height: | Size: 50 KiB | 
| After Width: | Height: | Size: 50 KiB | 
| After Width: | Height: | Size: 48 KiB | 
| After Width: | Height: | Size: 49 KiB | 
|  | @ -223,6 +223,11 @@ const tags = [ | |||
| 		slug: 'bisexual', | ||||
| 		priority: 10, | ||||
| 	}, | ||||
| 	{ | ||||
| 		name: 'black cock', | ||||
| 		slug: 'black-cock', | ||||
| 		group: 'body', | ||||
| 	}, | ||||
| 	{ | ||||
| 		name: 'black hair', | ||||
| 		slug: 'black-hair', | ||||
|  | @ -1022,6 +1027,11 @@ const tags = [ | |||
| 		priority: 7, | ||||
| 		group: 'ethnicity', | ||||
| 	}, | ||||
| 	{ | ||||
| 		name: 'white cock', | ||||
| 		slug: 'white-cock', | ||||
| 		group: 'body', | ||||
| 	}, | ||||
| 	{ | ||||
| 		name: 'wife', | ||||
| 		slug: 'wife', | ||||
|  |  | |||
|  | @ -607,9 +607,10 @@ const tagMedia = [ | |||
| 	['airtight', 9, 'Cindy Shine in GP1658'], | ||||
| 	['anal', 5, 'Abella Danger', 'hardx'], | ||||
| 	['anal', 'kira_noir_julesjordan', 'Kira Noir in "Kira Noir Opens Her Ass For Manuel"', 'julesjordan'], | ||||
| 	['anal', 7, 'Anastasia Brokelyn', 'bangbros'], | ||||
| 	['anal', 'jane_wilde_evilangel_2', 'Jane Wilde and Brock Cooper in "The Cock Hungry Chronicles"', 'evilangel'], | ||||
| 	['anal', 0, 'Adriana Chechik in "Manuel Creampies Their Asses 3"', 'julesjordan'], | ||||
| 	['anal', 'nikki_benz_bigwetbutts', 'Nikki Benz in "Pantyhose Playtime"', 'bigwetbutts'], | ||||
| 	['anal', 7, 'Anastasia Brokelyn', 'bangbros'], | ||||
| 	['anal', 6, 'Chloe Cherry in "Chloe\'s Big Anal"', 'darkx'], | ||||
| 	['anal', 4, 'Lana Roy in "Anal In The Club"', '21naturals'], | ||||
| 	['anal', 3, 'Dakota Skye', 'brazzers'], | ||||
|  | @ -650,6 +651,7 @@ const tagMedia = [ | |||
| 	['blonde', 2, 'Isabelle Deltore', 'herlimit'], | ||||
| 	['blowbang', 0, 'Lacy Lennon in "Lacy Lennon\'s First Blowbang"', 'hardx'], | ||||
| 	['blowbang', 'zaawaadi_roccosiffredi_1', 'Zaawaadi in "My Name Is Zaawaadi"', 'roccosiffredi'], | ||||
| 	['blowbang', 'gina_gerson_assholefever', 'Gina Gerson in "Oppa Gangbang Style"', 'assholefever'], | ||||
| 	['blowbang', 1, 'Nicole Black in GIO1680', 'legalporno'], | ||||
| 	['blowjob', 'clanddi_jinkcego_ddfbusty_1', 'Clanddi Jinkcego', 'ddfbusty'], | ||||
| 	['blowjob', 4, 'Chloe Cherry in "Chloe\'s Big Anal"', 'darkx'], | ||||
|  | @ -813,6 +815,7 @@ const tagMedia = [ | |||
| 	['fake-cum', 2, 'Mimi Allen', 'fuckedupfacials'], | ||||
| 	['fake-cum', 3, 'Alexia Anders in "Thanksgiving Creampies"', 'cum4k'], | ||||
| 	['fake-cum', 0, 'Jynx Maze in "Showering Slut Gets Sludged"', 'cumshotsurprise'], | ||||
| 	['fake-cum', 'franceska_le_fortyozbounce', 'Franceska Le and Luscious Lopez', 'fortyozbounce'], | ||||
| 	['fake-cum', 1, 'Ricki White', 'fuckedupfacials'], | ||||
| 	['fake-cum', 4, 'Vina Sky in "Creaming Her Pipes"', 'anal4k'], | ||||
| 	['family', 0, 'Teanna Trump in "A Family Appear: Part One"', 'brazzers'], | ||||
|  | @ -828,6 +831,7 @@ const tagMedia = [ | |||
| 	['gangbang', 'kristen_scott_julesjordan', 'Kristen Scott in "Interracial Gangbang!"', 'julesjordan'], | ||||
| 	['gangbang', 'lara_frost_legalporno_1', 'Lara Frost in NRX070', 'legalporno'], | ||||
| 	['gangbang', 7, 'Alexa Flexy in GL376', 'legalporno'], | ||||
| 	['gangbang', 'gina_gerson_assholefever', 'Gina Gerson in "Oppa Gangbang Style"', 'assholefever'], | ||||
| 	['gangbang', 0, '"4 On 1 Gangbangs"', 'doghousedigital'], | ||||
| 	['gangbang', 4, 'Marley Brinx in "The Gangbang of Marley Brinx"', 'julesjordan'], | ||||
| 	['gangbang', 1, 'Ginger Lynn in "Gangbang Mystique", a photoset shot by Suze Randall, 1984. Depicting a woman \'airtight\' pushed the boundaries of pornography at the time.'], | ||||
|  | @ -851,6 +855,7 @@ const tagMedia = [ | |||
| 	['lesbian', 0, 'Jenna Sativa and Alina Lopez in "Opposites Attract"', 'girlgirl'], | ||||
| 	['maid', 0, 'Whitney Wright in "Dredd Up Your Ass 2"', 'julesjordan'], | ||||
| 	['maid', 1, 'Alessandra Jane', 'brazzers'], | ||||
| 	['milf', 'nikki_benz_bigwetbutts', 'Nikki Benz in "Pantyhose Playtime"', 'bigwetbutts'], | ||||
| 	['milf', 'silvia_saige_ddfnetwork', 'Silvia Saige', 'pornworld'], | ||||
| 	['milf', 2, 'Shalina Devine', 'analmom'], | ||||
| 	['milf', 1, 'Francesca Le', 'evilangel'], | ||||
|  | @ -861,6 +866,7 @@ const tagMedia = [ | |||
| 	['mfm', 0, 'Vina Sky in "Jules Jordan\'s Three Ways"', 'julesjordan'], | ||||
| 	['mfm', 8, 'Ariana Marie in "DP Masters 7"', 'julesjordan'], | ||||
| 	['mfm', 1, 'Lana Rhoades in "Gangbang Me 3"', 'hardx'], | ||||
| 	['mfm', 'franceska_jaimes_digitalplayground', 'Franceska Jaimes in "Monarch"', 'digitalplayground'], | ||||
| 	['mfm', 'hazel_moore_legalporno', 'Hazel Moore', 'legalporno'], | ||||
| 	['mfm', 7, 'Rose Valerie', 'eurosexparties'], | ||||
| 	['mfm', 6, 'Honey Gold in "Slut Puppies 12"', 'julesjordan'], | ||||
|  | @ -940,6 +946,7 @@ const tagMedia = [ | |||
| 	['toys', 1, 'Chloe Lamour in "Curives In All The Right Places"', 'wetandpuffy'], | ||||
| 	['toys', 'shawna_lenee_sunrisekings', 'Shawna Lenee', 'sunrisekings'], | ||||
| 	['trainbang', 'poster', 'Kali Roses in "Passing Me Around"', 'blacked'], | ||||
| 	['trainbang', 'gina_gerson_assholefever', 'Gina Gerson in "Oppa Gangbang Style"', 'assholefever'], | ||||
| 	['vr', 0, 'Michelle H', 'metart'], | ||||
| 	['vr', '1a', 'Jenna Fox and Tommy Pistol in "Virtual Reality Jenna Fox Fucks So Real"', 'bangbros'], | ||||
| 	['white', 2, 'Kenzie Reeves', 'bang'], | ||||
|  |  | |||
|  | @ -0,0 +1,212 @@ | |||
| const bulkInsert = require('../src/utils/bulk-insert'); | ||||
| 
 | ||||
| const affiliates = { | ||||
| 	julesjordan_signup: { | ||||
| 		url: 'https://enter.julesjordan.com/track/Mzk3MS4yLjMuNi4wLjAuMC4wLjA', | ||||
| 		comment: '$30 per signup', | ||||
| 	}, | ||||
| 	manuelferrara_signup: { | ||||
| 		url: 'https://enter.manuelferrara.com/track/Mzk3MS4yLjcuMTYuMC4wLjAuMC4w', | ||||
| 		comment: '$30 per signup', | ||||
| 	}, | ||||
| 	spermswallowers_signup: { | ||||
| 		url: 'https://enter.spermswallowers.com/track/Mzk3MS4yLjUuMTMuMC4wLjAuMC4w', | ||||
| 		comment: '$30 per signup', | ||||
| 	}, | ||||
| 	theassfactory_signup: { | ||||
| 		url: 'https://enter.theassfactory.com/track/Mzk3MS4yLjEuMS4wLjAuMC4wLjA', | ||||
| 		comment: '$30 per signup', | ||||
| 	}, | ||||
| 	legalporno_new: { | ||||
| 		url: 'https://www.legalporno.com/new-videos?aff=BW90MHT1DP____', | ||||
| 		comment: 'default offer', | ||||
| 	}, | ||||
| 	pornworld_new: { | ||||
| 		url: 'https://pornworld.com/new-videos?aff=BW90MHT1DP____', | ||||
| 		comment: 'default offer', | ||||
| 	}, | ||||
| }; | ||||
| 
 | ||||
| const banners = [ | ||||
| 	{ | ||||
| 		id: 'julesjordan_728_90_jill_kassidy', | ||||
| 		width: 728, | ||||
| 		height: 90, | ||||
| 		network: 'julesjordan', | ||||
| 		tags: ['sex', 'blowjob', 'black-cock', 'brunette'], | ||||
| 	}, | ||||
| 	{ | ||||
| 		id: 'julesjordan_728_90_angela_white', | ||||
| 		width: 728, | ||||
| 		height: 90, | ||||
| 		network: 'julesjordan', | ||||
| 		tags: ['sex', 'black-cock', 'brunette'], | ||||
| 	}, | ||||
| 	{ | ||||
| 		id: 'julesjordan_728_90_adriana_chechik', | ||||
| 		width: 728, | ||||
| 		height: 90, | ||||
| 		network: 'julesjordan', | ||||
| 		tags: ['anal', 'black-cock', 'brunette'], | ||||
| 	}, | ||||
| 	{ | ||||
| 		id: 'julesjordan_728_90_autumn_falls', | ||||
| 		width: 728, | ||||
| 		height: 90, | ||||
| 		network: 'julesjordan', | ||||
| 		tags: ['sex', 'big-boobs', 'brunette'], | ||||
| 	}, | ||||
| 	{ | ||||
| 		id: 'julesjordan_728_90_gabbie_carter', | ||||
| 		width: 728, | ||||
| 		height: 90, | ||||
| 		network: 'julesjordan', | ||||
| 		tags: ['sex', 'blowjob', 'big-boobs', 'brunette'], | ||||
| 	}, | ||||
| 	{ | ||||
| 		id: 'manuelferrara_728_90_asses', | ||||
| 		width: 728, | ||||
| 		height: 90, | ||||
| 		channel: 'manuelferrara', | ||||
| 		tags: ['big-butt'], | ||||
| 	}, | ||||
| 	{ | ||||
| 		id: 'pornworld_600_120_1', | ||||
| 		width: 600, | ||||
| 		height: 120, | ||||
| 		network: 'pornworld', | ||||
| 		tags: ['anal', 'brunette'], | ||||
| 	}, | ||||
| 	{ | ||||
| 		id: 'pornworld_600_120_2', | ||||
| 		width: 600, | ||||
| 		height: 120, | ||||
| 		network: 'pornworld', | ||||
| 		tags: ['mfm', 'sex', 'brunette'], | ||||
| 	}, | ||||
| ]; | ||||
| 
 | ||||
| const links = [ | ||||
| 	{ | ||||
| 		network: 'julesjordan', | ||||
| 		affiliate: 'julesjordan_signup', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'julesjordan', | ||||
| 		affiliate: 'julesjordan_signup', | ||||
| 		banner: 'julesjordan_728_90_jill_kassidy', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'julesjordan', | ||||
| 		affiliate: 'julesjordan_signup', | ||||
| 		banner: 'julesjordan_728_90_angela_white', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'julesjordan', | ||||
| 		affiliate: 'julesjordan_signup', | ||||
| 		banner: 'julesjordan_728_90_adriana_chechik', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'julesjordan', | ||||
| 		affiliate: 'julesjordan_signup', | ||||
| 		banner: 'julesjordan_728_90_autumn_falls', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'julesjordan', | ||||
| 		affiliate: 'julesjordan_signup', | ||||
| 		banner: 'julesjordan_728_90_gabbie_carter', | ||||
| 	}, | ||||
| 	{ | ||||
| 		channel: 'manuelferrara', | ||||
| 		affiliate: 'manuelferrara_signup', | ||||
| 	}, | ||||
| 	{ | ||||
| 		channel: 'manuelferrara', | ||||
| 		affiliate: 'manuelferrara_signup', | ||||
| 		banner: 'manuelferrara_728_90_asses', | ||||
| 	}, | ||||
| 	{ | ||||
| 		channel: 'spermswallowers', | ||||
| 		affiliate: 'spermswallowers_signup', | ||||
| 	}, | ||||
| 	{ | ||||
| 		channel: 'theassfactory', | ||||
| 		affiliate: 'theassfactory_signup', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'legalporno', | ||||
| 		affiliate: 'legalporno_new', | ||||
| 	}, | ||||
| 	{ | ||||
| 		channel: 'legalporno', | ||||
| 		affiliate: 'legalporno_new', | ||||
| 		banner: 'pornworld_600_120_1', | ||||
| 	}, | ||||
| 	{ | ||||
| 		channel: 'legalporno', | ||||
| 		affiliate: 'legalporno_new', | ||||
| 		banner: 'pornworld_600_120_2', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'pornworld', | ||||
| 		affiliate: 'pornworld_new', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'pornworld', | ||||
| 		affiliate: 'pornworld_new', | ||||
| 		banner: 'pornworld_600_120_1', | ||||
| 	}, | ||||
| 	{ | ||||
| 		network: 'pornworld', | ||||
| 		affiliate: 'pornworld_new', | ||||
| 		banner: 'pornworld_600_120_2', | ||||
| 	}, | ||||
| ]; | ||||
| 
 | ||||
| exports.seed = async knex => Promise.resolve() | ||||
| 	.then(async () => { | ||||
| 		await Promise.all([ | ||||
| 			knex('campaigns').delete(), | ||||
| 			knex('affiliates').delete(), | ||||
| 			knex('banners').delete(), | ||||
| 		]); | ||||
| 
 | ||||
| 		await bulkInsert('affiliates', Object.entries(affiliates).map(([key, value]) => ({ id: key, ...value })), false); | ||||
| 
 | ||||
| 		const [networks, channels, tags] = await Promise.all([ | ||||
| 			knex('entities') | ||||
| 				.where('type', 'network') | ||||
| 				.whereIn('slug', links.concat(banners).map(link => link.network).filter(Boolean)), | ||||
| 			knex('entities') | ||||
| 				.where('type', 'channel') | ||||
| 				.whereIn('slug', links.concat(banners).map(link => link.channel).filter(Boolean)), | ||||
| 			knex('tags') | ||||
| 				.whereIn('slug', banners.flatMap(banner => banner.tags || [])), | ||||
| 		]); | ||||
| 
 | ||||
| 		const networksBySlug = networks.reduce((acc, network) => ({ ...acc, [network.slug]: network }), {}); | ||||
| 		const channelsBySlug = channels.reduce((acc, channel) => ({ ...acc, [channel.slug]: channel }), {}); | ||||
| 		const tagsBySlug = tags.reduce((acc, tag) => ({ ...acc, [tag.slug]: tag }), {}); | ||||
| 
 | ||||
| 		const linksWithEntityIdAndAffiliateId = links.map(link => ({ | ||||
| 			entity_id: networksBySlug[link.network]?.id || channelsBySlug[link.channel]?.id, | ||||
| 			affiliate_id: link.affiliate, | ||||
| 			banner_id: link.banner, | ||||
| 		})).filter(link => link.entity_id && link.affiliate_id); | ||||
| 
 | ||||
| 		const bannersWithEntityId = banners.map(banner => ({ | ||||
| 			id: banner.id, | ||||
| 			width: banner.width, | ||||
| 			height: banner.height, | ||||
| 			entity_id: networksBySlug[banner.network]?.id || channelsBySlug[banner.channel]?.id || null, | ||||
| 		})); | ||||
| 
 | ||||
| 		const bannerTags = banners.flatMap(banner => banner.tags?.map(tag => ({ | ||||
| 			banner_id: banner.id, | ||||
| 			tag_id: tagsBySlug[tag].id, | ||||
| 		})) || []); | ||||
| 
 | ||||
| 		await bulkInsert('banners', bannersWithEntityId, false); | ||||
| 		await bulkInsert('banners_tags', bannerTags, false); | ||||
| 		await bulkInsert('campaigns', linksWithEntityIdAndAffiliateId, false); | ||||
| 	}); | ||||
|  | @ -22,7 +22,7 @@ async function verifyPassword(password, storedPassword) { | |||
| } | ||||
| 
 | ||||
| async function login(credentials) { | ||||
| 	if (!config.auth.enabled) { | ||||
| 	if (!config.auth.login) { | ||||
| 		throw new HttpError('Authentication is disabled', 405); | ||||
| 	} | ||||
| 
 | ||||
|  | @ -38,7 +38,7 @@ async function login(credentials) { | |||
| } | ||||
| 
 | ||||
| async function signup(credentials) { | ||||
| 	if (!config.auth.enabled) { | ||||
| 	if (!config.auth.signup) { | ||||
| 		throw new HttpError('Authentication is disabled', 405); | ||||
| 	} | ||||
| 
 | ||||
|  |  | |||
|  | @ -152,7 +152,8 @@ async function initServer() { | |||
| 		res.render(path.join(__dirname, '../../assets/index.ejs'), { | ||||
| 			env: JSON.stringify({ | ||||
| 				sfw: !!req.headers.sfw || Object.prototype.hasOwnProperty.call(req.query, 'sfw'), | ||||
| 				auth: config.auth.enabled, | ||||
| 				login: config.auth.login, | ||||
| 				signup: config.auth.signup, | ||||
| 				sessionId: req.session.safeId, | ||||
| 			}), | ||||
| 		}); | ||||
|  |  | |||