Added Meiden van Holland and Vurig Vlaanderen.

This commit is contained in:
DebaucheryLibrarian 2023-07-01 21:46:44 +02:00
parent 01b3cc42af
commit 205102ff90
64 changed files with 508 additions and 19 deletions

View File

@ -65,6 +65,7 @@
</a> </a>
<ul class="bio nolist"> <ul class="bio nolist">
<!-- probably not a good idea
<li <li
v-if="actor.realName" v-if="actor.realName"
class="bio-item" class="bio-item"
@ -72,6 +73,7 @@
<dfn class="bio-label"><Icon icon="vcard" />Real name</dfn> <dfn class="bio-label"><Icon icon="vcard" />Real name</dfn>
<span class="bio-value">{{ actor.realName }}</span> <span class="bio-value">{{ actor.realName }}</span>
</li> </li>
-->
<li <li
v-if="actor.dateOfBirth" v-if="actor.dateOfBirth"
@ -109,6 +111,15 @@
>{{ actor.ageAtDeath }}</span></span> >{{ actor.ageAtDeath }}</span></span>
</li> </li>
<li
v-if="actor.orientation"
class="bio-item"
>
<dfn class="bio-label"><Icon icon="heart7" />Orientation</dfn>
<span class="orientation">{{ actor.orientation }}</span>
</li>
<li <li
v-if="actor.origin" v-if="actor.origin"
class="bio-item birth" class="bio-item birth"
@ -294,7 +305,7 @@
class="description" class="description"
> >
{{ description.text }} {{ description.text }}
<router-link :to="`/${description.entity.type}/${description.entity.slug}`"> <RouterLink :to="`/${description.entity.type}/${description.entity.slug}`">
<img <img
v-if="description.entity.type === 'network' || !description.entity.parent || description.entity.independent" v-if="description.entity.type === 'network' || !description.entity.parent || description.entity.independent"
:src="`/img/logos/${description.entity.slug}/thumbs/network.png`" :src="`/img/logos/${description.entity.slug}/thumbs/network.png`"
@ -306,7 +317,7 @@
:src="`/img/logos/${description.entity.parent.slug}/thumbs/${description.entity.slug}.png`" :src="`/img/logos/${description.entity.parent.slug}/thumbs/${description.entity.slug}.png`"
class="description-logo" class="description-logo"
> >
</router-link> </RouterLink>
</p> </p>
</div> </div>
</div> </div>
@ -685,7 +696,8 @@ export default {
.ethnicity, .ethnicity,
.hair, .hair,
.eyes { .eyes,
.orientation {
text-transform: capitalize; text-transform: capitalize;
} }

View File

@ -53,6 +53,7 @@ function initActorActions(store, router) {
slug slug
realName realName
gender gender
orientation
dateOfBirth dateOfBirth
dateOfDeath dateOfDeath
age age

View File

@ -70,6 +70,7 @@ const actorFields = `
ageFromBirth ageFromBirth
dateOfBirth dateOfBirth
gender gender
orientation
avatar: avatarMedia { avatar: avatarMedia {
id id
path path

View File

@ -0,0 +1,19 @@
exports.up = async (knex) => {
await knex.schema.alterTable('actors', (table) => {
table.text('orientation');
});
await knex.schema.alterTable('actors_profiles', (table) => {
table.text('orientation');
});
};
exports.down = async (knex) => {
await knex.schema.alterTable('actors', (table) => {
table.dropColumn('orientation');
});
await knex.schema.alterTable('actors_profiles', (table) => {
table.dropColumn('orientation');
});
};

14
package-lock.json generated
View File

@ -79,7 +79,7 @@
"tunnel": "0.0.6", "tunnel": "0.0.6",
"ua-parser-js": "^1.0.32", "ua-parser-js": "^1.0.32",
"undici": "^4.13.0", "undici": "^4.13.0",
"unprint": "^0.9.3", "unprint": "^0.9.4",
"url-pattern": "^1.0.3", "url-pattern": "^1.0.3",
"v-tooltip": "^2.0.3", "v-tooltip": "^2.0.3",
"video.js": "^7.11.4", "video.js": "^7.11.4",
@ -17512,9 +17512,9 @@
} }
}, },
"node_modules/unprint": { "node_modules/unprint": {
"version": "0.9.3", "version": "0.9.4",
"resolved": "https://registry.npmjs.org/unprint/-/unprint-0.9.3.tgz", "resolved": "https://registry.npmjs.org/unprint/-/unprint-0.9.4.tgz",
"integrity": "sha512-ujDlQL0yeVVd6V+kN5uURG/6F9jUblF0VWMOcpI9u3ZjsWp2tC4mQy0/kK4epU8QhkEFPE9uZ0pAMKORzEdp5g==", "integrity": "sha512-KU5hU99TUQPoXBu7E6oUMPOiCzQK70oHmHRnKLaCvQHpkU+q8opzl3bsNLzs4tTUcyWn8FNIdhRS1++bmwqrWg==",
"dependencies": { "dependencies": {
"axios": "^0.27.2", "axios": "^0.27.2",
"bottleneck": "^2.19.5", "bottleneck": "^2.19.5",
@ -32339,9 +32339,9 @@
"integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
}, },
"unprint": { "unprint": {
"version": "0.9.3", "version": "0.9.4",
"resolved": "https://registry.npmjs.org/unprint/-/unprint-0.9.3.tgz", "resolved": "https://registry.npmjs.org/unprint/-/unprint-0.9.4.tgz",
"integrity": "sha512-ujDlQL0yeVVd6V+kN5uURG/6F9jUblF0VWMOcpI9u3ZjsWp2tC4mQy0/kK4epU8QhkEFPE9uZ0pAMKORzEdp5g==", "integrity": "sha512-KU5hU99TUQPoXBu7E6oUMPOiCzQK70oHmHRnKLaCvQHpkU+q8opzl3bsNLzs4tTUcyWn8FNIdhRS1++bmwqrWg==",
"requires": { "requires": {
"axios": "^0.27.2", "axios": "^0.27.2",
"bottleneck": "^2.19.5", "bottleneck": "^2.19.5",

View File

@ -138,7 +138,7 @@
"tunnel": "0.0.6", "tunnel": "0.0.6",
"ua-parser-js": "^1.0.32", "ua-parser-js": "^1.0.32",
"undici": "^4.13.0", "undici": "^4.13.0",
"unprint": "^0.9.3", "unprint": "^0.9.4",
"url-pattern": "^1.0.3", "url-pattern": "^1.0.3",
"v-tooltip": "^2.0.3", "v-tooltip": "^2.0.3",
"video.js": "^7.11.4", "video.js": "^7.11.4",

Binary file not shown.

After

Width:  |  Height:  |  Size: 145 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 80 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 39 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 28 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 87 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 63 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 33 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 33 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 19 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 19 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.0 KiB

View File

@ -198,6 +198,16 @@ const tags = [
name: 'behind the scenes', name: 'behind the scenes',
slug: 'bts', slug: 'bts',
}, },
{
name: 'belgian',
slug: 'belgian',
group: 'ethnicity',
},
{
name: 'flemish',
slug: 'flemish',
group: 'ethnicity',
},
{ {
name: 'big dick', name: 'big dick',
slug: 'big-dick', slug: 'big-dick',
@ -359,6 +369,10 @@ const tags = [
name: 'curvy', name: 'curvy',
slug: 'curvy', slug: 'curvy',
}, },
{
name: 'debut',
slug: 'debut',
},
{ {
name: 'doctor', name: 'doctor',
slug: 'doctor', slug: 'doctor',
@ -720,6 +734,11 @@ const tags = [
slug: 'milf', slug: 'milf',
group: 'age', group: 'age',
}, },
{
name: 'mature',
slug: 'mature',
group: 'age',
},
{ {
name: 'MFM threesome', name: 'MFM threesome',
slug: 'mfm', slug: 'mfm',
@ -859,6 +878,14 @@ const tags = [
name: 'roleplay', name: 'roleplay',
slug: 'roleplay', slug: 'roleplay',
}, },
{
name: 'erotic',
slug: 'erotic',
},
{
name: 'romantic',
slug: 'romantic',
},
{ {
name: 'rough', name: 'rough',
slug: 'rough', slug: 'rough',
@ -1119,6 +1146,14 @@ const tags = [
slug: 'office', slug: 'office',
group: 'location', group: 'location',
}, },
{
name: 'young',
slug: 'young',
},
{
name: 'exotic',
slug: 'exotic',
},
]; ];
const aliases = [ const aliases = [
@ -1142,10 +1177,22 @@ const aliases = [
name: '3+ on 1', name: '3+ on 1',
for: 'gangbang', for: 'gangbang',
}, },
{
name: 'trio 2 meiden 1 man',
for: 'mff',
},
{
name: 'trio 1 meid 2 mannen',
for: 'mfm',
},
{ {
name: 'anal sex', name: 'anal sex',
for: 'anal', for: 'anal',
}, },
{
name: 'anale sex',
for: 'anal',
},
{ {
name: 'anal drill', name: 'anal drill',
for: 'anal', for: 'anal',
@ -1499,6 +1546,10 @@ const aliases = [
name: 'pussy licking', name: 'pussy licking',
for: 'pussy-eating', for: 'pussy-eating',
}, },
{
name: 'debutanten',
for: 'debut',
},
{ {
name: 'deep throat', name: 'deep throat',
for: 'deepthroat', for: 'deepthroat',
@ -1512,6 +1563,10 @@ const aliases = [
name: 'dildo', name: 'dildo',
for: 'toys', for: 'toys',
}, },
{
name: 'dikke tieten',
for: 'big-boobs',
},
{ {
name: 'doggystyle', name: 'doggystyle',
for: 'doggy-style', for: 'doggy-style',
@ -1740,7 +1795,7 @@ const aliases = [
}, },
{ {
name: 'gilf', name: 'gilf',
for: 'older-men', for: 'mature',
}, },
{ {
name: 'girl girl', name: 'girl girl',
@ -1799,6 +1854,10 @@ const aliases = [
name: 'jerk-off instructions', name: 'jerk-off instructions',
for: 'joi', for: 'joi',
}, },
{
name: 'jonge meid',
for: 'young',
},
{ {
name: 'lashing', name: 'lashing',
for: 'corporal-punishment', for: 'corporal-punishment',
@ -2142,6 +2201,10 @@ const aliases = [
name: 'caucasian', name: 'caucasian',
for: 'white', for: 'white',
}, },
{
name: 'vlaams',
for: 'flemish',
},
{ {
name: 'work', name: 'work',
for: 'office', for: 'office',
@ -2154,6 +2217,66 @@ const aliases = [
name: 'zapper', name: 'zapper',
for: 'electric-shock', for: 'electric-shock',
}, },
{
name: 'amateur sex',
for: 'amateur',
},
{
name: 'lesbische sex',
for: 'lesbian',
},
{
name: 'neuken 1 op 1',
for: 'sex',
},
{
name: 'exotische meid',
for: 'exotic',
},
{
name: 'rollenspel',
for: 'roleplay',
},
{
name: 'blondje',
for: 'blonde',
},
{
name: 'rossig',
for: 'redhead',
},
{
name: 'grote-billen',
for: 'big-butt',
},
{
name: 'interraciaal',
for: 'interracial',
},
{
name: 'squirten',
for: 'squirting',
},
{
name: 'buitensex',
for: 'outdoors',
},
{
name: 'volle vrouw',
for: 'curvy',
},
{
name: 'romantische sex',
for: 'romantic',
},
{
name: 'pijpbeurt',
for: 'blowjob',
},
{
name: 'pijpen',
for: 'blowjob',
},
]; ];
const priorities = [ // higher index is higher priority const priorities = [ // higher index is higher priority

View File

@ -169,6 +169,11 @@ const networks = [
}, },
parent: 'gamma', parent: 'gamma',
}, },
{
slug: 'bluedonkeymedia',
name: 'Blue Donkey Media',
url: 'https://www.blowpass.com',
},
{ {
slug: 'brazzers', slug: 'brazzers',
name: 'Brazzers', name: 'Brazzers',

View File

@ -1975,6 +1975,41 @@ const sites = [
description: '', description: '',
parent: 'blowpass', parent: 'blowpass',
}, },
// BLUE DONKEY MEDIA
{
slug: 'meidenvanholland',
name: 'Meiden Van Holland',
url: 'https://meidenvanholland.nl',
independent: true,
parameters: {
frontend: 1,
languages: ['nl', 'en'],
secret: 'HwzAace4HlS1W9h4YdPLhG6OzZjFCmIy6LUYhfv3',
},
parent: 'bluedonkeymedia',
},
{
slug: 'vurigvlaanderen',
name: 'Vurig Vlaanderen',
url: 'https://vurigvlaanderen.be',
independent: true,
parameters: {
frontend: 3,
languages: ['be'],
secret: 'DYRQhSPLpGcTaPumqyvXT3KvEvTQ2LUKmpvYdUjX',
},
parent: 'bluedonkeymedia',
},
{
slug: 'ashleymore',
name: 'Ashley More',
url: 'https://ashleymore.nl',
parameters: {
frontend: 19,
secret: 'bNKswWRMTpsjoawpoFBOETAfPZzoyrnPIpbyjyrn',
},
parent: 'bluedonkeymedia',
},
// BRAD MONTANA // BRAD MONTANA
{ {
slug: 'bradmontana', slug: 'bradmontana',

View File

@ -40,30 +40,50 @@ const hairColors = {
'soft-black': 'black', 'soft-black': 'black',
'brunette/raven': 'brown', 'brunette/raven': 'brown',
black: 'black', black: 'black',
blond: 'blond',
blonde: 'blonde', blonde: 'blonde',
blondie: 'blonde', blondie: 'blonde',
brown: 'brown', brown: 'brown',
bruin: 'brown',
brunette: 'brown', brunette: 'brown',
fair: 'blonde', fair: 'blonde',
raven: 'black', raven: 'black',
red: 'red', red: 'red',
redhead: 'red', redhead: 'red',
rood: 'red',
blue: 'blue', blue: 'blue',
green: 'green', green: 'green',
purple: 'purple', purple: 'purple',
pink: 'pink', pink: 'pink',
zwart: 'black',
}; };
const eyeColors = { const eyeColors = {
blauw: 'blue',
blue: 'blue', blue: 'blue',
brown: 'brown', brown: 'brown',
bruin: 'bruin',
dark: 'brown', dark: 'brown',
gray: 'gray', gray: 'gray',
green: 'green', green: 'green',
groen: 'green',
grey: 'gray', grey: 'gray',
hazel: 'hazel', hazel: 'hazel',
}; };
const orientations = {
bi: 'bisexual',
biseksueel: 'bisexual',
bisexual: 'bisexual',
gay: 'gay',
hetero: 'straight',
heteroseksueel: 'straight',
heterosexual: 'straight',
homoseksueel: 'gay',
homosexual: 'gay',
straight: 'straight',
};
const ethnicities = { const ethnicities = {
'african american': 'black', 'african american': 'black',
'african-american': 'black', 'african-american': 'black',
@ -205,6 +225,7 @@ function curateActor(actor, withDetails = false, isProfile = false) {
slug: actor.slug, slug: actor.slug,
url: actor.url, url: actor.url,
gender: actor.gender, gender: actor.gender,
orientation: actor.orientation,
entityId: actor.entity_id, entityId: actor.entity_id,
aliasFor: actor.alias_for, aliasFor: actor.alias_for,
dateOfBirth: actor.date_of_birth, dateOfBirth: actor.date_of_birth,
@ -303,6 +324,7 @@ function curateProfileEntry(profile) {
age: profile.age, age: profile.age,
url: profile.url, url: profile.url,
gender: profile.gender, gender: profile.gender,
orientation: profile.orientation,
ethnicity: profile.ethnicity, ethnicity: profile.ethnicity,
description: profile.description, description: profile.description,
description_hash: profile.descriptionHash, description_hash: profile.descriptionHash,
@ -372,6 +394,8 @@ async function curateProfile(profile, actor) {
|| (/male/i.test(profile.gender) && 'male') || (/male/i.test(profile.gender) && 'male')
|| null; || null;
curatedProfile.orientation = orientations[profile.orientation?.trim()] || null;
const dateOfBirth = profile.dateOfBirth || profile.birthdate; const dateOfBirth = profile.dateOfBirth || profile.birthdate;
curatedProfile.dateOfBirth = (!Number.isNaN(Number(dateOfBirth)) // possibly valid date curatedProfile.dateOfBirth = (!Number.isNaN(Number(dateOfBirth)) // possibly valid date
@ -528,6 +552,7 @@ async function interpolateProfiles(actorIdsOrNames) {
const mostFrequentValues = [ const mostFrequentValues = [
'gender', 'gender',
'orientation',
'ethnicity', 'ethnicity',
'cup', 'cup',
'bust', 'bust',
@ -669,6 +694,7 @@ async function scrapeProfiles(actor, sources, entitiesBySlug, existingProfilesBy
...entity, ...entity,
// legacy // legacy
site: entity, site: entity,
channel: entity,
network: entity?.parent, network: entity?.parent,
entity, entity,
scraper: scraperSlug, scraper: scraperSlug,

View File

@ -56,12 +56,15 @@ function toBaseReleases(baseReleasesOrUrls, entity = null) {
} }
async function fetchUnprintScene(scraper, url, entity, baseRelease, options, type) { async function fetchUnprintScene(scraper, url, entity, baseRelease, options, type) {
const releaseScraper = scraper[type === 'movie' ? 'scrapeMovie' : 'scrapeScene'];
const res = await unprint.get(url, { const res = await unprint.get(url, {
rejectUnauthorized: false, rejectUnauthorized: false,
...(releaseScraper.scraper && releaseScraper), // options object
}); });
if (res.ok) { if (res.ok) {
return scraper[type === 'movie' ? 'scrapeMovie' : 'scrapeScene'](res.context, { return (releaseScraper.scraper || releaseScraper)(res.context, {
url, url,
entity, entity,
baseRelease, baseRelease,
@ -78,7 +81,7 @@ async function fetchScene(scraper, url, entity, baseRelease, options, type = 'sc
} }
if ((type === 'scene' && scraper.scrapeScene) || (type === 'movie' && scraper.scrapeMovie)) { if ((type === 'scene' && scraper.scrapeScene) || (type === 'movie' && scraper.scrapeMovie)) {
if (scraper.useUnprint) { if (scraper.useUnprint || scraper.scrapeScene?.unprint || scraper.scrapeMovie?.unprint) {
return fetchUnprintScene(scraper, url, entity, baseRelease, options, type); return fetchUnprintScene(scraper, url, entity, baseRelease, options, type);
} }

View File

@ -163,7 +163,7 @@ function toBaseSource(rawSource) {
return null; return null;
} }
if (rawSource.match(/.m3u8$/)) { if (new URL(rawSource).pathname.match(/.m3u8$/)) {
return { return {
src: rawSource, src: rawSource,
stream: rawSource, stream: rawSource,
@ -175,6 +175,12 @@ function toBaseSource(rawSource) {
}; };
} }
if (typeof rawSource === 'function') {
return {
defer: rawSource,
};
}
return null; return null;
} }
@ -266,12 +272,12 @@ async function findSourceDuplicates(baseMedias) {
.filter(Boolean); .filter(Boolean);
const extractUrls = baseMedias const extractUrls = baseMedias
.map((baseMedia) => baseMedia.sources.map((source) => source.url)) .map((baseMedia) => baseMedia.sources.map((source) => source.extract))
.flat() .flat()
.filter(Boolean); .filter(Boolean);
const [existingSourceMedia, existingExtractMedia] = await Promise.all([ const [existingSourceMedia, existingExtractMedia] = await Promise.all([
// my try to check thousands of URLs at once, don't pass all of them to a single query // may try to check thousands of URLs at once, don't pass all of them to a single query
chunk(sourceUrls).reduce(async (chain, sourceUrlsChunk) => { chunk(sourceUrls).reduce(async (chain, sourceUrlsChunk) => {
const accUrls = await chain; const accUrls = await chain;
const existingUrls = await knex('media').whereIn('source', sourceUrlsChunk); const existingUrls = await knex('media').whereIn('source', sourceUrlsChunk);
@ -344,6 +350,17 @@ async function findHashDuplicates(medias) {
} }
async function extractSource(baseSource, { existingExtractMediaByUrl }) { async function extractSource(baseSource, { existingExtractMediaByUrl }) {
if (typeof baseSource.defer === 'function') {
const src = await baseSource.defer();
console.log('DEFERED', src);
return {
...baseSource,
...toBaseSource(src),
};
}
if (typeof baseSource.extract !== 'function' || !baseSource.url) { if (typeof baseSource.extract !== 'function' || !baseSource.url) {
return baseSource; return baseSource;
} }
@ -365,7 +382,7 @@ async function extractSource(baseSource, { existingExtractMediaByUrl }) {
return { return {
...baseSource, ...baseSource,
src, ...toBaseSource(src),
}; };
} }

View File

@ -0,0 +1,242 @@
'use strict';
const crypto = require('crypto');
const unprint = require('unprint');
const http = require('../utils/http');
async function fetchTrailer(entryId, videoId, channel, credentials) {
const url = `https://api.sysero.nl/free-stream?resource_id=${entryId}&video_id=${videoId}`;
const res = await http.get(url, {
headers: {
Origin: channel.url,
Credentials: credentials,
},
});
if (res.ok) {
return res.body.data?.attributes.sources.streams.mpd?.url;
}
return null;
}
// MVH's slug system seems to break on non-alphanumerical characters, but also supports ID
function getSceneUrl(channel, slug, sceneId) {
if (slug && /^[\w-]+$/i.test(slug)) {
return `${channel.url}/sexfilms/${slug}`;
}
return `${channel.url}/sexfilms/${sceneId}`;
}
function scrapeAll(scenes, channel, context) {
return scenes.reduce((acc, scene) => {
const release = {};
release.entryId = scene.id;
release.url = getSceneUrl(channel, scene.attributes.slug, scene.id);
release.date = unprint.extractDate(scene.attributes.product.active_from, 'D/M/YY');
release.title = scene.attributes.title;
release.description = scene.attributes.description;
release.duration = unprint.extractDuration(scene.attributes.videos.film?.[0]?.duration);
const posterPath = scene.attributes.images.thumb?.[0]?.path || context.images[scene.id];
const teaserPath = context.clips[scene.relationships.clips?.data[0]?.id];
if (posterPath) {
release.poster = `https://cdndo.sysero.nl${scene.attributes.images.thumb?.[0]?.path || context.images[scene.id]}`;
}
if (scene.attributes.videos.trailer?.[0]) {
release.trailer = async () => fetchTrailer(scene.id, scene.attributes.videos.trailer[0].id, channel, context.credentials);
}
if (teaserPath) {
release.teaser = `https://cdndo.sysero.nl${teaserPath}`;
}
release.tags = scene.relationships.categories?.data.map((category) => context.tags[category.id]?.replace(/-/g, ' ')).filter(Boolean);
release.language = scene.attributes.videos.film?.[0]?.language;
if (release.language && channel.parameters.languages && !channel.parameters.languages?.includes(release.language)) {
// all MVH sites list the entire network, but we want to store Flemish scenes under Vurig Vlaanderen
return { ...acc, unextracted: [...acc.unextracted, release] };
}
return { ...acc, scenes: [...acc.scenes, release] };
}, {
scenes: [],
unextracted: [],
});
}
function getCredentials(channel) {
const now = Math.floor(Date.now() / 1000);
const hash = crypto
.createHmac('sha256', channel.parameters.secret)
.update(`${channel.parameters.frontend}${now.toString()}`)
.digest('hex');
const credentials = `Syserauth ${channel.parameters.frontend}-${hash}-${now.toString(16)}`;
return credentials;
}
const falseCountry = /afghanistan/i; // no country defaults to Afghanistan
function getLocation(model) {
const country = model.country && !falseCountry.test(model.country) ? model.country : null;
return [model.city, model.county, country]
.map((segment) => segment?.trim())
.filter(Boolean)
.join(', ') || null;
}
function scrapeProfile(model, { entity, includeScenes = true }) {
const actor = {};
actor.name = model.title;
actor.url = unprint.prefixUrl(`/modellen/${model.slug}`, entity.url);
actor.entryId = model.id;
actor.description = model.description;
actor.dateOfBirth = model.birth_date && model.age > 18 ? new Date(model.birth_date) : null; // sometimes seems to be profile creation date
actor.age = model.age > 18 ? model.age : null;
actor.orientation = model.sexual_orientation;
actor.birthPlace = getLocation(model);
actor.height = Number(model.length) || null;
actor.weight = Number(model.weight) || null;
actor.eyes = model.eye_color;
actor.hairColor = model.hair_color;
if (includeScenes) {
actor.scenes = model.videos?.map((video) => ({
entryId: video.id,
url: getSceneUrl(entity, video.slug, video.id),
title: video.title,
description: video.description,
}));
}
actor.avatar = unprint.prefixUrl(model.images?.[0]?.path, 'https://cdndo.sysero.nl');
return actor;
}
function scrapeSceneData(scene, { entity }) {
const release = {};
release.entryId = scene.id;
release.url = getSceneUrl(entity, scene.slug, scene.id);
release.title = scene.title;
release.description = scene.description;
release.date = scene.uploadDate
? new Date(scene.uploadDate)
: unprint.extractDate(scene.product.active_from, 'D/M/YY');
release.actors = scene.models?.map((model) => scrapeProfile(model, { entity, includeScenes: false }));
release.duration = scene.seconds || unprint.extractTimestamp(scene.isoDuration) || Number(scene.video_paid?.duration) * 60;
release.tags = scene.categories?.map((category) => category.slug.replace(/-/g, ' '));
if (scene.thumb) {
release.poster = [
scene.thumb.original,
scene.thumb.xxl,
scene.thumb.xl,
// ... l, m, s, xs, xxs, probably little point trying all of them
].map((poster) => unprint.prefixUrl(poster, 'https://cdndo.sysero.nl'));
}
release.photos = scene.gallery;
if (scene.trailer) {
release.trailer = async () => {
const credentials = getCredentials(entity);
return fetchTrailer(scene.id, scene.trailer.id, entity, credentials);
};
}
return release;
}
function scrapeScene({ _query, window }, context) {
const data = window.__NUXT__?.state?.videoStore?.video;
if (data) {
return scrapeSceneData(data, context);
}
return null;
}
async function fetchLatest(channel, page, context) {
const credentials = getCredentials(channel);
const res = await http.get(`https://api.sysero.nl/videos?page=${page}&count=20&type=video&include=images:types(thumb|thumb_mobile),categories,clips&filter[status]=published&filter[products]=1%2C2&sort[published_at]=DESC&frontend=${channel.parameters.frontend}`, {
headers: {
Origin: channel.url,
Credentials: credentials,
},
});
if (res.ok && res.body.data) {
const tags = Object.fromEntries(res.body.included?.filter((item) => item.type === 'category').map((item) => [item.id, item.attributes.slug]) || []);
const images = Object.fromEntries(res.body.included?.filter((item) => item.type === 'image' && item.attributes.types === 'thumb').map((item) => [item.id, item.attributes.path]) || []);
const clips = Object.fromEntries(res.body.included?.filter((item) => item.type === 'clip').map((item) => [item.id, item.attributes.path]) || []);
return scrapeAll(res.body.data, channel, { ...context, images, clips, tags, credentials });
}
return res.status;
}
async function fetchProfile(actor, { entity }) {
const credentials = getCredentials(entity);
const url = `${entity.url}/modellen/${actor.slug}`;
const res = await unprint.get(url, {
headers: {
Origin: entity.url,
Credentials: credentials,
},
parser: {
runScripts: 'dangerously',
},
});
if (res.ok) {
const data = res.context.window.__NUXT__?.state?.modelStore?.model;
if (data) {
return scrapeProfile(data, { entity });
}
return null;
}
return res.status;
}
module.exports = {
fetchLatest,
fetchProfile,
scrapeScene: {
scraper: scrapeScene,
unprint: true,
parser: {
runScripts: 'dangerously',
},
},
};

View File

@ -38,6 +38,7 @@ const kink = require('./kink');
const analvids = require('./analvids'); const analvids = require('./analvids');
const littlecapricedreams = require('./littlecapricedreams'); const littlecapricedreams = require('./littlecapricedreams');
const loveherfilms = require('./loveherfilms'); const loveherfilms = require('./loveherfilms');
const bluedonkeymedia = require('./bluedonkeymedia');
const mikeadriano = require('./mikeadriano'); const mikeadriano = require('./mikeadriano');
const mindgeek = require('./mindgeek'); const mindgeek = require('./mindgeek');
const naughtyamerica = require('./naughtyamerica'); const naughtyamerica = require('./naughtyamerica');
@ -125,6 +126,7 @@ const scrapers = {
littlecapricedreams, littlecapricedreams,
loveherfilms, loveherfilms,
mamacitaz: porndoe, mamacitaz: porndoe,
bluedonkeymedia,
mikeadriano, mikeadriano,
mindgeek, mindgeek,
mylf: teamskeet, mylf: teamskeet,
@ -187,6 +189,9 @@ const scrapers = {
blackedraw: vixen, blackedraw: vixen,
blackambush: elevatedx, blackambush: elevatedx,
blowpass, blowpass,
bluedonkeymedia,
meidenvanholland: bluedonkeymedia,
vurigvlaanderen: bluedonkeymedia,
boobpedia, boobpedia,
bradmontana, bradmontana,
brattysis: nubiles, brattysis: nubiles,

View File

@ -43,7 +43,7 @@ const defaultOptions = {
encodeJSON: true, encodeJSON: true,
parse: false, parse: false,
headers: { headers: {
'user-agent': 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36', 'user-agent': 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36',
}, },
}; };