Added movie tile. Fixed actor header. Larger breakpoint for nav menu.
|
@ -29,7 +29,7 @@
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="actor-inner">
|
<div class="content-inner actor-inner">
|
||||||
<div
|
<div
|
||||||
class="profile"
|
class="profile"
|
||||||
:class="{ expanded: bioExpanded, 'with-avatar': !!actor.avatar }"
|
:class="{ expanded: bioExpanded, 'with-avatar': !!actor.avatar }"
|
||||||
|
@ -714,8 +714,13 @@ export default {
|
||||||
margin: 1rem 0 0 0;
|
margin: 1rem 0 0 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.actor-header {
|
||||||
|
padding: .5rem 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
.header-name {
|
.header-name {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
font-size: 1.3rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -356,7 +356,7 @@ export default {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media(max-width: $breakpoint-micro) {
|
@media(max-width: $breakpoint) {
|
||||||
.nav {
|
.nav {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,48 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="movies">
|
<div class="movies">
|
||||||
<h1 class="heading">Movies</h1>
|
<div class="tiles">
|
||||||
|
<Movie
|
||||||
|
v-for="movie in movies"
|
||||||
|
:key="`movie-${movie.id}`"
|
||||||
|
:movie="movie"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import Movie from './tile.vue';
|
||||||
|
|
||||||
|
async function mounted() {
|
||||||
|
const { movies, totalCount } = await this.$store.dispatch('fetchMovies', {
|
||||||
|
limit: 30,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.movies = movies;
|
||||||
|
this.totalCount = totalCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
components: {
|
||||||
|
Movie,
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
movies: [],
|
||||||
|
totalCount: 0,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
mounted,
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
.movies {
|
.movies {
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tiles {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fill, 12rem);
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -0,0 +1,38 @@
|
||||||
|
<template>
|
||||||
|
<div class="tile">
|
||||||
|
<div class="details">{{ movie.entity.name }}</div>
|
||||||
|
<h3 class="title">{{ movie.title }}</h3>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
movie: {
|
||||||
|
type: Object,
|
||||||
|
default: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.tile {
|
||||||
|
background: var(--background);
|
||||||
|
box-shadow: 0 0 3px var(--darken);
|
||||||
|
}
|
||||||
|
|
||||||
|
.details {
|
||||||
|
color: var(--text-light);
|
||||||
|
background: var(--profile);
|
||||||
|
padding: .5rem 1rem;
|
||||||
|
font-size: .8rem;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
padding: 1rem;
|
||||||
|
margin: 0;
|
||||||
|
font-size: 1rem;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -48,13 +48,11 @@ async function mounted() {
|
||||||
'creampie',
|
'creampie',
|
||||||
'squirting',
|
'squirting',
|
||||||
],
|
],
|
||||||
ethnicity: [
|
appearance: [
|
||||||
'asian',
|
'asian',
|
||||||
'ebony',
|
'ebony',
|
||||||
'latina',
|
'latina',
|
||||||
'caucasian',
|
'caucasian',
|
||||||
],
|
|
||||||
appearance: [
|
|
||||||
'natural-boobs',
|
'natural-boobs',
|
||||||
'fake-boobs',
|
'fake-boobs',
|
||||||
'blonde',
|
'blonde',
|
||||||
|
|
|
@ -3,12 +3,6 @@
|
||||||
v-if="tag.poster"
|
v-if="tag.poster"
|
||||||
class="tile"
|
class="tile"
|
||||||
>
|
>
|
||||||
<router-link
|
|
||||||
class="title"
|
|
||||||
:to="{ name: 'tag', params: { tagSlug: tag.slug, range: 'latest' } }"
|
|
||||||
:title="tag.name"
|
|
||||||
>{{ tag.name }}</router-link>
|
|
||||||
|
|
||||||
<router-link
|
<router-link
|
||||||
:to="{ name: 'tag', params: { tagSlug: tag.slug, range: 'latest' } }"
|
:to="{ name: 'tag', params: { tagSlug: tag.slug, range: 'latest' } }"
|
||||||
:title="tag.name"
|
:title="tag.name"
|
||||||
|
@ -48,6 +42,13 @@
|
||||||
class="poster"
|
class="poster"
|
||||||
>
|
>
|
||||||
</router-link>
|
</router-link>
|
||||||
|
|
||||||
|
<router-link
|
||||||
|
class="title"
|
||||||
|
:to="{ name: 'tag', params: { tagSlug: tag.slug, range: 'latest' } }"
|
||||||
|
:title="tag.name"
|
||||||
|
>{{ tag.name }}</router-link>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<span
|
<span
|
||||||
|
@ -85,17 +86,12 @@ export default {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
position: relative;
|
position: relative;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
|
font-size: 0;
|
||||||
|
|
||||||
&:hover {
|
&:hover .poster {
|
||||||
.title {
|
|
||||||
color: var(--primary);
|
|
||||||
}
|
|
||||||
|
|
||||||
.poster {
|
|
||||||
box-shadow: 0 0 3px var(--darken);
|
box-shadow: 0 0 3px var(--darken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
.poster {
|
.poster {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
@ -103,13 +99,15 @@ export default {
|
||||||
}
|
}
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
display: inline-block;
|
display: block;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
padding: .25rem 1rem .25rem .5rem;
|
padding: .5rem;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
color: var(--shadow-strong);
|
color: var(--text-light);
|
||||||
|
background: var(--profile);
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
|
font-size: .9rem;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
text-transform: capitalize;
|
text-transform: capitalize;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
|
|
|
@ -7,8 +7,6 @@
|
||||||
|
|
||||||
<title>traxxx</title>
|
<title>traxxx</title>
|
||||||
|
|
||||||
<link rel="stylesheet" href="/css/style.css">
|
|
||||||
|
|
||||||
<link rel="apple-touch-icon" sizes="180x180" href="/img/favicon/apple-touch-icon.png">
|
<link rel="apple-touch-icon" sizes="180x180" href="/img/favicon/apple-touch-icon.png">
|
||||||
<link rel="icon" type="image/png" sizes="32x32" href="/img/favicon/favicon-32x32.png">
|
<link rel="icon" type="image/png" sizes="32x32" href="/img/favicon/favicon-32x32.png">
|
||||||
<link rel="icon" type="image/png" sizes="16x16" href="/img/favicon/favicon-16x16.png">
|
<link rel="icon" type="image/png" sizes="16x16" href="/img/favicon/favicon-16x16.png">
|
||||||
|
@ -18,6 +16,8 @@
|
||||||
<meta name="msapplication-TileColor" content="#aa2c66">
|
<meta name="msapplication-TileColor" content="#aa2c66">
|
||||||
<meta name="msapplication-config" content="/img/favicon/browserconfig.xml">
|
<meta name="msapplication-config" content="/img/favicon/browserconfig.xml">
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="/css/style.css">
|
||||||
|
|
||||||
<script src="/js/bundle.js" defer></script>
|
<script src="/js/bundle.js" defer></script>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
|
|
|
@ -180,6 +180,13 @@ function initEntitiesActions(store, _router) {
|
||||||
type: {
|
type: {
|
||||||
equalTo: "network"
|
equalTo: "network"
|
||||||
}
|
}
|
||||||
|
childEntitiesConnection: {
|
||||||
|
some: {
|
||||||
|
type: {
|
||||||
|
equalTo: "channel"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
independent: {
|
independent: {
|
||||||
|
@ -197,7 +204,7 @@ function initEntitiesActions(store, _router) {
|
||||||
in: $entitySlugs
|
in: $entitySlugs
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
]
|
],
|
||||||
}
|
}
|
||||||
) {
|
) {
|
||||||
id
|
id
|
||||||
|
|
|
@ -243,11 +243,6 @@ const releaseFragment = `
|
||||||
${releaseTrailerFragment}
|
${releaseTrailerFragment}
|
||||||
${releaseTeaserFragment}
|
${releaseTeaserFragment}
|
||||||
${siteFragment}
|
${siteFragment}
|
||||||
scenes: releasesMoviesByMovieId {
|
|
||||||
scene {
|
|
||||||
${releaseFields}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
studio {
|
studio {
|
||||||
id
|
id
|
||||||
name
|
name
|
||||||
|
|
|
@ -47,9 +47,48 @@ function initReleasesActions(store, _router) {
|
||||||
return curateRelease(release);
|
return curateRelease(release);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function fetchMovies({ _commit }, { limit = 10, pageNumber = 1 }) {
|
||||||
|
const { connection: { movies, totalCount } } = await graphql(`
|
||||||
|
query Movies(
|
||||||
|
$limit:Int = 1000,
|
||||||
|
$offset:Int = 0,
|
||||||
|
) {
|
||||||
|
connection: moviesConnection(
|
||||||
|
first: $limit
|
||||||
|
offset: $offset
|
||||||
|
orderBy: DATE_ASC
|
||||||
|
) {
|
||||||
|
movies: nodes {
|
||||||
|
id
|
||||||
|
title
|
||||||
|
url
|
||||||
|
slug
|
||||||
|
date
|
||||||
|
datePrecision
|
||||||
|
entity {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
slug
|
||||||
|
}
|
||||||
|
}
|
||||||
|
totalCount
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`, {
|
||||||
|
limit,
|
||||||
|
offset: Math.max(0, (pageNumber - 1)) * limit,
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
movies: movies.map(release => curateRelease(release)),
|
||||||
|
totalCount,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
fetchReleases,
|
fetchReleases,
|
||||||
fetchReleaseById,
|
fetchReleaseById,
|
||||||
|
fetchMovies,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -643,6 +643,32 @@ exports.up = knex => Promise.resolve()
|
||||||
table.datetime('created_at')
|
table.datetime('created_at')
|
||||||
.defaultTo(knex.fn.now());
|
.defaultTo(knex.fn.now());
|
||||||
}))
|
}))
|
||||||
|
.then(() => knex.schema.createTable('movies_covers', (table) => {
|
||||||
|
table.integer('movie_id', 16)
|
||||||
|
.notNullable()
|
||||||
|
.references('id')
|
||||||
|
.inTable('movies');
|
||||||
|
|
||||||
|
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)
|
||||||
|
.notNullable()
|
||||||
|
.references('id')
|
||||||
|
.inTable('movies');
|
||||||
|
|
||||||
|
table.text('media_id', 21)
|
||||||
|
.notNullable()
|
||||||
|
.references('id')
|
||||||
|
.inTable('media');
|
||||||
|
|
||||||
|
table.unique('movie_id');
|
||||||
|
}))
|
||||||
.then(() => knex.schema.createTable('releases', (table) => {
|
.then(() => knex.schema.createTable('releases', (table) => {
|
||||||
table.increments('id', 16);
|
table.increments('id', 16);
|
||||||
|
|
||||||
|
@ -965,6 +991,9 @@ exports.down = (knex) => { // eslint-disable-line arrow-body-style
|
||||||
DROP TABLE IF EXISTS releases_tags CASCADE;
|
DROP TABLE IF EXISTS releases_tags CASCADE;
|
||||||
DROP TABLE IF EXISTS releases_search CASCADE;
|
DROP TABLE IF EXISTS releases_search CASCADE;
|
||||||
|
|
||||||
|
DROP TABLE IF EXISTS movies_covers CASCADE;
|
||||||
|
DROP TABLE IF EXISTS movies_trailers CASCADE;
|
||||||
|
|
||||||
DROP TABLE IF EXISTS batches CASCADE;
|
DROP TABLE IF EXISTS batches CASCADE;
|
||||||
|
|
||||||
DROP TABLE IF EXISTS actors_avatars CASCADE;
|
DROP TABLE IF EXISTS actors_avatars CASCADE;
|
||||||
|
|
Before Width: | Height: | Size: 684 KiB After Width: | Height: | Size: 348 KiB |
Before Width: | Height: | Size: 8.2 KiB After Width: | Height: | Size: 8.2 KiB |
Before Width: | Height: | Size: 46 KiB After Width: | Height: | Size: 36 KiB |
Before Width: | Height: | Size: 38 KiB After Width: | Height: | Size: 30 KiB |
Before Width: | Height: | Size: 992 KiB After Width: | Height: | Size: 1.3 MiB |
Before Width: | Height: | Size: 8.0 KiB After Width: | Height: | Size: 7.6 KiB |
Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 34 KiB |
Before Width: | Height: | Size: 41 KiB After Width: | Height: | Size: 31 KiB |
Before Width: | Height: | Size: 38 KiB After Width: | Height: | Size: 30 KiB |
|
@ -7920,7 +7920,7 @@ const sites = [
|
||||||
{
|
{
|
||||||
name: 'PornDoe Pedia',
|
name: 'PornDoe Pedia',
|
||||||
slug: 'porndoepedia',
|
slug: 'porndoepedia',
|
||||||
url: 'https://vipsexvault.com/channels/vipsexvault-pedia.en.html',
|
url: 'https://vipsexvault.com/channels/porndoe-pedia.en.html',
|
||||||
parent: 'vipsexvault',
|
parent: 'vipsexvault',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
|
@ -615,7 +615,7 @@ const tagPosters = [
|
||||||
['dvp', 'poster', 'Riley Reid in "Pizza That Ass" for Reid My Lips'],
|
['dvp', 'poster', 'Riley Reid in "Pizza That Ass" for Reid My Lips'],
|
||||||
['dv-tp', 'poster', 'Juelz Ventura in "Gangbanged 5" for Elegant Angel'],
|
['dv-tp', 'poster', 'Juelz Ventura in "Gangbanged 5" for Elegant Angel'],
|
||||||
['ebony', 2, 'Nia Nacci for Sweetheart Video'],
|
['ebony', 2, 'Nia Nacci for Sweetheart Video'],
|
||||||
['facefucking', 2, 'Jynx Maze for Throated'],
|
['facefucking', 1, 'Paige Owens in "Dark Meat 12" for Evil Angel'],
|
||||||
['facial', 0, 'Brooklyn Gray in "All About Ass 4" for Evil Angel'],
|
['facial', 0, 'Brooklyn Gray in "All About Ass 4" for Evil Angel'],
|
||||||
['fake-boobs', 2, 'Gia Milana in "Hot Anal Latina" for HardX'],
|
['fake-boobs', 2, 'Gia Milana in "Hot Anal Latina" for HardX'],
|
||||||
['family', 0, 'Teanna Trump in "A Family Appear: Part One" for Brazzers'],
|
['family', 0, 'Teanna Trump in "A Family Appear: Part One" for Brazzers'],
|
||||||
|
@ -703,8 +703,8 @@ const tagPhotos = [
|
||||||
['ebony', 1, 'Ana Foxxx in "DP Me 4" for HardX'],
|
['ebony', 1, 'Ana Foxxx in "DP Me 4" for HardX'],
|
||||||
['facial', 2, 'Ashly Anderson for Hookup Hotshot'],
|
['facial', 2, 'Ashly Anderson for Hookup Hotshot'],
|
||||||
['facial', 'poster', 'Jynx Maze'],
|
['facial', 'poster', 'Jynx Maze'],
|
||||||
|
['facefucking', 2, 'Jynx Maze for Throated'],
|
||||||
['facefucking', 3, 'Adriana Chechik in "Performing Magic Butt Tricks With Jules Jordan. What Will Disappear In Her Ass?" for Jules Jordan'],
|
['facefucking', 3, 'Adriana Chechik in "Performing Magic Butt Tricks With Jules Jordan. What Will Disappear In Her Ass?" for Jules Jordan'],
|
||||||
['facefucking', 1, 'Carrie for Young Throats'],
|
|
||||||
['fake-boobs', 7, 'Madison Ivy for Passion HD'],
|
['fake-boobs', 7, 'Madison Ivy for Passion HD'],
|
||||||
['fake-boobs', 1, 'Lela Star in "Thick" for Jules Jordan'],
|
['fake-boobs', 1, 'Lela Star in "Thick" for Jules Jordan'],
|
||||||
['fake-boobs', 6, 'Cathy Heaven in "Heavenly Ass" for Big Wett Butts'],
|
['fake-boobs', 6, 'Cathy Heaven in "Heavenly Ass" for Big Wett Butts'],
|
||||||
|
|
21
src/app.js
|
@ -8,7 +8,7 @@ const initServer = require('./web/server');
|
||||||
const knex = require('./knex');
|
const knex = require('./knex');
|
||||||
const fetchUpdates = require('./updates');
|
const fetchUpdates = require('./updates');
|
||||||
const { fetchScenes, fetchMovies } = require('./deep');
|
const { fetchScenes, fetchMovies } = require('./deep');
|
||||||
const { storeReleases, updateReleasesSearch } = require('./store-releases');
|
const { storeReleases, storeMovies, updateReleasesSearch } = require('./store-releases');
|
||||||
const { scrapeActors } = require('./actors');
|
const { scrapeActors } = require('./actors');
|
||||||
const getFileEntries = require('./utils/file-entries');
|
const getFileEntries = require('./utils/file-entries');
|
||||||
|
|
||||||
|
@ -31,14 +31,14 @@ async function init() {
|
||||||
const updateBaseScenes = (argv.all || argv.channels || argv.networks || argv.movies) && await fetchUpdates();
|
const updateBaseScenes = (argv.all || argv.channels || argv.networks || argv.movies) && await fetchUpdates();
|
||||||
|
|
||||||
const scenesFromFile = argv.scenesFile && await getFileEntries(argv.scenesFile);
|
const scenesFromFile = argv.scenesFile && await getFileEntries(argv.scenesFile);
|
||||||
const sceneUrls = (argv.scenes || []).concat(scenesFromFile || []);
|
const sceneUrls = (argv.scene || []).concat(scenesFromFile || []);
|
||||||
|
|
||||||
const deepScenes = argv.deep
|
const deepScenes = argv.deep
|
||||||
? await fetchScenes([...(sceneUrls), ...(updateBaseScenes || []), ...(actorBaseScenes || [])])
|
? await fetchScenes([...(sceneUrls), ...(updateBaseScenes || []), ...(actorBaseScenes || [])])
|
||||||
: [...(updateBaseScenes || []), ...(actorBaseScenes || [])];
|
: [...(updateBaseScenes || []), ...(actorBaseScenes || [])];
|
||||||
|
|
||||||
const sceneMovies = deepScenes && argv.sceneMovies && deepScenes.map(scene => scene.movie).filter(Boolean);
|
const sceneMovies = deepScenes && argv.movie && deepScenes.map(scene => scene.movie).filter(Boolean);
|
||||||
const deepMovies = await fetchMovies([...(argv.movies || []), ...(sceneMovies || [])]);
|
const deepMovies = await fetchMovies([...(argv.movie || []), ...(sceneMovies || [])]);
|
||||||
|
|
||||||
if (argv.inspect) {
|
if (argv.inspect) {
|
||||||
console.log(util.inspect(deepScenes));
|
console.log(util.inspect(deepScenes));
|
||||||
|
@ -46,10 +46,15 @@ async function init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (argv.save) {
|
if (argv.save) {
|
||||||
await storeReleases([
|
if (deepScenes.length > 0) {
|
||||||
...(deepScenes || []),
|
await storeReleases(deepScenes);
|
||||||
...(deepMovies || []),
|
}
|
||||||
]);
|
|
||||||
|
console.log(deepMovies);
|
||||||
|
|
||||||
|
if (deepMovies.length > 0) {
|
||||||
|
await storeMovies(deepMovies);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
knex.destroy();
|
knex.destroy();
|
||||||
|
|
20
src/argv.js
|
@ -25,10 +25,6 @@ const { argv } = yargs
|
||||||
type: 'array',
|
type: 'array',
|
||||||
alias: 'channel',
|
alias: 'channel',
|
||||||
})
|
})
|
||||||
.option('movies', {
|
|
||||||
describe: 'Scrape movies from channels',
|
|
||||||
type: 'array',
|
|
||||||
})
|
|
||||||
.option('actors', {
|
.option('actors', {
|
||||||
describe: 'Scrape actors by name or slug',
|
describe: 'Scrape actors by name or slug',
|
||||||
type: 'array',
|
type: 'array',
|
||||||
|
@ -61,19 +57,18 @@ const { argv } = yargs
|
||||||
alias: 'with-profiles',
|
alias: 'with-profiles',
|
||||||
default: false,
|
default: false,
|
||||||
})
|
})
|
||||||
.option('scenes', {
|
.option('scene', {
|
||||||
describe: 'Scrape scene info from URL',
|
describe: 'Scrape scene info from URL',
|
||||||
type: 'array',
|
type: 'array',
|
||||||
alias: 'scene',
|
|
||||||
})
|
})
|
||||||
.option('scenes-file', {
|
.option('scene-file', {
|
||||||
describe: 'Scrape scene info from URLs in a file',
|
describe: 'Scrape scene info from URLs in a file',
|
||||||
type: 'string',
|
type: 'string',
|
||||||
|
alias: 'scenes-file',
|
||||||
})
|
})
|
||||||
.option('movie', {
|
.option('movie', {
|
||||||
describe: 'Scrape movie info from URL',
|
describe: 'Scrape movie info from URL',
|
||||||
type: 'array',
|
type: 'array',
|
||||||
alias: 'movies',
|
|
||||||
})
|
})
|
||||||
.option('sources', {
|
.option('sources', {
|
||||||
describe: 'Use these scrapers for actor data',
|
describe: 'Use these scrapers for actor data',
|
||||||
|
@ -88,12 +83,17 @@ const { argv } = yargs
|
||||||
.option('latest', {
|
.option('latest', {
|
||||||
describe: 'Scrape latest releases if available',
|
describe: 'Scrape latest releases if available',
|
||||||
type: 'boolean',
|
type: 'boolean',
|
||||||
default: true,
|
default: false,
|
||||||
})
|
})
|
||||||
.option('upcoming', {
|
.option('upcoming', {
|
||||||
describe: 'Scrape upcoming releases if available',
|
describe: 'Scrape upcoming releases if available',
|
||||||
type: 'boolean',
|
type: 'boolean',
|
||||||
default: true,
|
default: false,
|
||||||
|
})
|
||||||
|
.option('movies', {
|
||||||
|
describe: 'Scrape movies from channels',
|
||||||
|
type: 'boolean',
|
||||||
|
default: false,
|
||||||
})
|
})
|
||||||
.option('force', {
|
.option('force', {
|
||||||
describe: 'Don\'t ignore duplicates, update existing entries',
|
describe: 'Don\'t ignore duplicates, update existing entries',
|
||||||
|
|
|
@ -612,7 +612,7 @@ async function storeMedias(baseMedias) {
|
||||||
return [...newMediaWithEntries, ...existingHashMedias];
|
return [...newMediaWithEntries, ...existingHashMedias];
|
||||||
}
|
}
|
||||||
|
|
||||||
async function associateReleaseMedia(releases) {
|
async function associateReleaseMedia(releases, type = 'releases') {
|
||||||
if (!argv.media) {
|
if (!argv.media) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -670,7 +670,7 @@ async function associateReleaseMedia(releases) {
|
||||||
.filter(Boolean);
|
.filter(Boolean);
|
||||||
|
|
||||||
if (associations.length > 0) {
|
if (associations.length > 0) {
|
||||||
await knex.raw(`${knex(`releases_${role}`).insert(associations)} ON CONFLICT DO NOTHING`);
|
await knex.raw(`${knex(`${type}_${role}`).insert(associations)} ON CONFLICT DO NOTHING`);
|
||||||
}
|
}
|
||||||
}, Promise.resolve());
|
}, Promise.resolve());
|
||||||
}
|
}
|
||||||
|
|
|
@ -276,13 +276,13 @@ async function scrapeScene({ html, qu }, url, site, include) {
|
||||||
return release;
|
return release;
|
||||||
}
|
}
|
||||||
|
|
||||||
function scrapeMovie({ el, qu }, url, site) {
|
function scrapeMovie({ el, query }, url, site) {
|
||||||
const movie = { url, site };
|
const movie = { url, site };
|
||||||
|
|
||||||
movie.entryId = qu.q('.dvd_details_overview .rating_box').dataset.id;
|
movie.entryId = query.q('.rating_box').dataset.id;
|
||||||
movie.title = qu.q('.title_bar span', true);
|
movie.title = query.q('.title_bar span', true);
|
||||||
movie.covers = qu.urls('#dvd-cover-flip > a');
|
movie.covers = query.urls('#dvd-cover-flip > a');
|
||||||
movie.channel = slugify(qu.q('.update_date a', true), '');
|
movie.channel = slugify(query.q('.update_date a', true), '');
|
||||||
|
|
||||||
// movie.releases = Array.from(document.querySelectorAll('.cell.dvd_info > a'), el => el.href);
|
// movie.releases = Array.from(document.querySelectorAll('.cell.dvd_info > a'), el => el.href);
|
||||||
const sceneQus = ctxa(el, '.dvd_details');
|
const sceneQus = ctxa(el, '.dvd_details');
|
||||||
|
|
|
@ -12,7 +12,7 @@ const { associateReleaseTags } = require('./tags');
|
||||||
const { curateEntity } = require('./entities');
|
const { curateEntity } = require('./entities');
|
||||||
const { associateReleaseMedia } = require('./media');
|
const { associateReleaseMedia } = require('./media');
|
||||||
|
|
||||||
function curateReleaseEntry(release, batchId, existingRelease) {
|
function curateReleaseEntry(release, batchId, existingRelease, type = 'scene') {
|
||||||
const slugBase = release.title
|
const slugBase = release.title
|
||||||
|| (release.actors?.length && `${release.entity.slug} ${release.actors.map(actor => actor.name).join(' ')}`)
|
|| (release.actors?.length && `${release.entity.slug} ${release.actors.map(actor => actor.name).join(' ')}`)
|
||||||
|| (release.date && `${release.entity.slug} ${formatDate(release.date, 'YYYY MM DD')}`)
|
|| (release.date && `${release.entity.slug} ${formatDate(release.date, 'YYYY MM DD')}`)
|
||||||
|
@ -23,19 +23,18 @@ function curateReleaseEntry(release, batchId, existingRelease) {
|
||||||
limit: config.titleSlugLength,
|
limit: config.titleSlugLength,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
console.log(release);
|
||||||
|
|
||||||
const curatedRelease = {
|
const curatedRelease = {
|
||||||
title: release.title,
|
title: release.title,
|
||||||
entry_id: release.entryId || null,
|
entry_id: release.entryId || null,
|
||||||
entity_id: release.entity.id,
|
entity_id: release.entity.id,
|
||||||
studio_id: release.studio?.id || null,
|
studio_id: release.studio?.id || null,
|
||||||
shoot_id: release.shootId || null,
|
|
||||||
url: release.url,
|
url: release.url,
|
||||||
date: Number(release.date) ? release.date : null,
|
date: Number(release.date) ? release.date : null,
|
||||||
production_date: Number(release.productionDate) ? release.productionDate : null,
|
|
||||||
date_precision: release.datePrecision,
|
date_precision: release.datePrecision,
|
||||||
slug,
|
slug,
|
||||||
description: release.description,
|
description: release.description,
|
||||||
duration: release.duration,
|
|
||||||
comment: release.comment,
|
comment: release.comment,
|
||||||
// director: release.director,
|
// director: release.director,
|
||||||
// likes: release.rating && release.rating.likes,
|
// likes: release.rating && release.rating.likes,
|
||||||
|
@ -46,6 +45,12 @@ function curateReleaseEntry(release, batchId, existingRelease) {
|
||||||
updated_batch_id: batchId,
|
updated_batch_id: batchId,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (type === 'scene') {
|
||||||
|
curatedRelease.shoot_id = release.shootId || null;
|
||||||
|
curatedRelease.productionDate = Number(release.productionDate) ? release.productionDate : null;
|
||||||
|
curatedRelease.duration = release.duration;
|
||||||
|
}
|
||||||
|
|
||||||
if (!existingRelease && !release.id) {
|
if (!existingRelease && !release.id) {
|
||||||
curatedRelease.created_batch_id = batchId;
|
curatedRelease.created_batch_id = batchId;
|
||||||
}
|
}
|
||||||
|
@ -252,7 +257,24 @@ async function storeReleases(releases) {
|
||||||
return releasesWithId;
|
return releasesWithId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function storeMovies(movies) {
|
||||||
|
const [batchId] = await knex('batches').insert({ comment: null }).returning('id');
|
||||||
|
|
||||||
|
console.log(movies);
|
||||||
|
|
||||||
|
const curatedMovieEntries = movies.map(release => curateReleaseEntry(release, batchId, null, 'movie'));
|
||||||
|
console.log(curatedMovieEntries);
|
||||||
|
const storedMovies = await knex.batchInsert('movies', curatedMovieEntries).returning('*');
|
||||||
|
|
||||||
|
const moviesWithId = attachReleaseIds(movies, storedMovies);
|
||||||
|
|
||||||
|
await associateReleaseMedia(moviesWithId, 'movies');
|
||||||
|
|
||||||
|
return storedMovies;
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
storeReleases,
|
storeReleases,
|
||||||
|
storeMovies,
|
||||||
updateReleasesSearch,
|
updateReleasesSearch,
|
||||||
};
|
};
|
||||||
|
|
|
@ -161,6 +161,21 @@ async function scrapeUpcomingReleases(scraper, entity, preData) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function scrapeMovies(scraper, entity) {
|
||||||
|
if (!scraper.fetchMovies) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// return await scrapeReleases(scraper, entity, preData, true);
|
||||||
|
return await scraper.fetchMovies(entity);
|
||||||
|
} catch (error) {
|
||||||
|
logger.warn(`Failed to scrape movies for '${entity.slug}' (${entity.parent?.slug}): ${error.message}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
async function scrapeChannelReleases(scraper, channelEntity, preData) {
|
async function scrapeChannelReleases(scraper, channelEntity, preData) {
|
||||||
const [latestReleases, upcomingReleases] = await Promise.all([
|
const [latestReleases, upcomingReleases] = await Promise.all([
|
||||||
argv.latest
|
argv.latest
|
||||||
|
@ -169,12 +184,11 @@ async function scrapeChannelReleases(scraper, channelEntity, preData) {
|
||||||
argv.upcoming
|
argv.upcoming
|
||||||
? scrapeUpcomingReleases(scraper, channelEntity, preData)
|
? scrapeUpcomingReleases(scraper, channelEntity, preData)
|
||||||
: [],
|
: [],
|
||||||
|
argv.movies
|
||||||
|
? scrapeMovies(scraper, channelEntity, preData)
|
||||||
|
: [],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
if (scraper.fetchMovies) {
|
|
||||||
await scraper.fetchMovies(channelEntity);
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.info(`Fetching ${latestReleases.length} latest and ${upcomingReleases.length} upcoming updates for '${channelEntity.name}' (${channelEntity.parent?.name})`);
|
logger.info(`Fetching ${latestReleases.length} latest and ${upcomingReleases.length} upcoming updates for '${channelEntity.name}' (${channelEntity.parent?.name})`);
|
||||||
|
|
||||||
return [...latestReleases, ...upcomingReleases];
|
return [...latestReleases, ...upcomingReleases];
|
||||||
|
|