Refactoring flow. Added user and profile saving and variables.

This commit is contained in:
ThePendulum 2018-04-22 23:46:14 +02:00
parent a3f0ad41ea
commit dc3f3c8440
9 changed files with 227 additions and 91 deletions

71
app.js
View File

@ -2,20 +2,53 @@
const config = require('config'); const config = require('config');
const util = require('util'); const util = require('util');
const fs = require('fs-extra');
const yargs = require('yargs').argv; const yargs = require('yargs').argv;
const snoowrap = require('snoowrap'); const snoowrap = require('snoowrap');
const curateSubmissions = require('./curate/submissions.js');
const curateUser = require('./curate/user.js');
const methods = require('./methods/methods.js'); const methods = require('./methods/methods.js');
const curate = require('./curate.js'); const interpolate = require('./interpolate.js');
const fetchItem = require('./fetchItem.js');
const fetchContent = require('./fetchContent.js'); const fetchContent = require('./fetchContent.js');
const save = require('./save.js');
const textToStream = require('./textToStream.js');
const reddit = new snoowrap(config.reddit.api); const reddit = new snoowrap(config.reddit.api);
function getSubmissions(users) { function saveProfileDetails(user) {
if(config.library.profile.image) {
// pass profile image as item to interpolate extension variable
const filepath = interpolate(config.library.profile.image, user, null, {
url: user.profile.image
});
fetchItem(user.profile.image).then(stream => save(filepath, stream)).catch(error => {
console.log('\x1b[33m%s\x1b[0m', `Could not save profile image for '${user.name}': ${error}`);
});
}
if(config.library.profile.description) {
if(user.profile.description) {
const filepath = interpolate(config.library.profile.description, user);
const stream = textToStream(user.profile.description);
save(filepath, stream).catch(error => {
console.log('\x1b[33m%s\x1b[0m', `Could not save profile description for '${user.name}': ${error}`);
});
} else {
console.log('\x1b[33m%s\x1b[0m', `No profile description for '${user.name}'`);
}
}
};
function getSubmissions(users, sort, limit) {
return users.reduce((chain, user) => { return users.reduce((chain, user) => {
return chain.then(acc => { return chain.then(acc => {
return reddit.getUser(user).getSubmissions({ return reddit.getUser(user).getSubmissions({
sort: yargs.sort || config.reddit.sort, sort: sort,
limit: yargs.limit || config.reddit.limit limit: limit
}).then(submissions => { }).then(submissions => {
return acc.concat(submissions); return acc.concat(submissions);
}); });
@ -23,11 +56,36 @@ function getSubmissions(users) {
}, Promise.resolve([])); }, Promise.resolve([]));
}; };
if(!yargs.user && typeof yargs.users !== 'string') {
return console.log('\x1b[31m%s\x1b[0m', 'Please supply at least one user with --user=[user], or multiple users with --users=[user1,user2] or --user=[user1] --user=[user2]');
}
const users = yargs.users ? yargs.users.split(',') : [].concat(yargs.user);
users.forEach(user => {
return Promise.resolve().then(() => {
// get reddit profile
return reddit.getUser(user).fetch().then(curateUser);
}).then(user => {
return saveProfileDetails(user);
// get submissions
}).catch(error => {
return console.log('\x1b[33m%s\x1b[0m', error);
});
});
/*
Promise.resolve().then(() => { Promise.resolve().then(() => {
if(yargs.user || yargs.users) { if(yargs.user || yargs.users) {
const users = yargs.users ? yargs.users.split(',') : [].concat(yargs.user); const users = yargs.users ? yargs.users.split(',') : [].concat(yargs.user);
return getSubmissions(users); return Promise.resolve().then(() => {
if(config.library.profile) {
return getProfiles(users);
}
}).then(() => {
return getSubmissions(users, yargs.sort || config.reddit.sort, yargs.limit === undefined ? config.reddit.limit : yargs.limit);
});
} }
return Promise.reject('Please supply at least one user with one or multiple --user, or --users!'); return Promise.reject('Please supply at least one user with one or multiple --user, or --users!');
@ -48,5 +106,6 @@ Promise.resolve().then(() => {
}).then(posts => { }).then(posts => {
return fetchContent(posts); return fetchContent(posts);
}).catch(error => { }).catch(error => {
return console.log('\x1b[33m%s\x1b[0m', error); return console.log('\x1b[31m%s\x1b[0m', error);
}); });
*/

View File

@ -1,18 +1,29 @@
module.exports = { module.exports = {
patterns: { library: {
image: 'output/$postUser/$postDate - $itemId - $postTitle$ext', base: 'output/$user/',
video: 'output/$postUser/$postDate - $itemId - $postTitle$ext', image: '$postDate - $itemId - $postTitle$ext',
text: 'output/$postUser/$postDate - $postId - $postTitle', video: '$postDate - $itemId - $postTitle$ext',
text: '$postDate - $postId - $postTitle',
album: { album: {
extractSingleItem: true, extractSingleItem: true,
image: 'output/$postUser/$postDate - $albumId - $postTitle/$itemIndex - $itemId$ext', image: '$postDate - $albumId - $postTitle/$itemIndex - $itemId$ext',
video: 'output/$postUser/$postDate - $albumId - $postTitle/$itemIndex - $itemId$ext' video: '$postDate - $albumId - $postTitle/$itemIndex - $itemId$ext'
},
profile: {
image: 'profile$ext',
description: 'profile ($userVerified$userVerifiedEmail$userGold)'
},
booleans: {
extracted: 'extracted-',
verified: '✔',
verifiedEmail: '✉',
gold: '★',
over18: '♥'
}, },
dateFormat: 'YYYYMMDD', dateFormat: 'YYYYMMDD',
titleLength: 200, titleLength: 200,
indexOffset: 1, indexOffset: 1,
slashSubstitute: '#', slashSubstitute: '#',
extractedLabel: 'extracted-'
}, },
reddit: { reddit: {
sort: 'top', sort: 'top',

View File

@ -1,8 +1,8 @@
'use strict'; 'use strict';
const dissectLink = require('./dissectLink.js'); const dissectLink = require('../dissectLink.js');
function curate(submissions) { function curateSubmissions(submissions) {
return submissions.map((submission, index) => { return submissions.map((submission, index) => {
return { return {
id: submission.id, id: submission.id,
@ -19,4 +19,4 @@ function curate(submissions) {
}); });
}; };
module.exports = curate; module.exports = curateSubmissions;

26
curate/user.js Normal file
View File

@ -0,0 +1,26 @@
'use strict';
const path = require('path');
function curateUser(user) {
console.log(user);
return {
id: user.id,
name: user.name,
created: new Date(user.created_utc * 1000),
gold: user.is_gold,
verified: user.verified,
verifiedEmail: user.has_verified_email,
profile: {
id: user.subreddit.display_name.name,
title: user.subreddit.display_name.title,
image: user.subreddit.display_name.icon_img,
banner: user.subreddit.display_name.banner_img,
description: user.subreddit.display_name.public_description,
over18: user.subreddit.display_name.over_18
}
};
};
module.exports = curateUser;

View File

@ -1,58 +1,13 @@
'use strict'; 'use strict';
const Readable = require('stream').Readable;
const fs = require('fs-extra'); const fs = require('fs-extra');
const path = require('path'); const path = require('path');
const config = require('config'); const config = require('config');
const fetch = require('node-fetch');
const fetchItem = require('./fetchItem');
const save = require('./save.js');
const interpolate = require('./interpolate.js'); const interpolate = require('./interpolate.js');
const textToStream = require('./textToStream.js');
function saveItemToDisk(stream, filepath) {
const file = fs.createWriteStream(filepath);
return new Promise((resolve, reject) => {
stream.pipe(file).on('error', error => {
reject(error);
}).on('finish', () => {
console.log(`Saved '${filepath}'`);
resolve(filepath);
});
});
};
function textPostToStream(item) {
const stream = new Readable();
stream.push(item.text);
stream.push(null);
return Object.assign(item, {
stream: stream
});
};
function fetchItem(item, post, attempt) {
function retry(error) {
console.log(error);
if(attempt < 3) {
console.log('Retrying...');
return fetchItem(item, post, ++attempt);
}
};
return fetch(item.url).then(res => {
return res.ok ? res : Promise.reject(`Failed to fetch ${item.url}`);
}).then(res => {
console.log(`Fetched '${item.url}'`);
return Object.assign({}, item, {
stream: res.body
});
}).catch(retry);
};
module.exports = function(posts) { module.exports = function(posts) {
return Promise.all(posts.map(post => { return Promise.all(posts.map(post => {
@ -61,20 +16,22 @@ module.exports = function(posts) {
item.index = index; item.index = index;
if(item.self) { if(item.self) {
return textPostToStream(item); return Object.assign({}, item, {stream: textToStream(item.text)});
} }
return fetchItem(item, post, 0); return fetchItem(item.url, 0).then(stream => {
return Object.assign({}, item, {stream});
});
})); }));
}).then(items => { }).then(items => {
return Promise.all(items.map(item => { return Promise.all(items.map(item => {
const type = item.type.split('/')[0]; const type = item.type.split('/')[0];
const filepath = post.content.album ? interpolate(config.patterns.album[type], post, item) : interpolate(config.patterns[type], post, item); const filepath = post.content.album ? interpolate(config.library.album[type], post.user, post, item) : interpolate(config.library[type], post.user, post, item);
return Promise.resolve().then(() => { return Promise.resolve().then(() => {
return fs.ensureDir(path.dirname(filepath)); return fs.ensureDir(path.dirname(filepath));
}).then(() => { }).then(() => {
return saveItemToDisk(item.stream, filepath) return save(filepath, item.stream)
}); });
})); }));
}); });

25
fetchItem.js Normal file
View File

@ -0,0 +1,25 @@
'use strict';
const fetch = require('node-fetch');
function fetchItem(url, attempt) {
function retry(error) {
console.log(error);
if(attempt < 3) {
console.log('Retrying...');
return fetchItem(url, ++attempt);
}
};
return fetch(url).then(res => {
return res.ok ? res : Promise.reject(`Failed to fetch ${url}`);
}).then(res => {
console.log(`Fetched '${url}'`);
return res.body;
}).catch(retry);
};
module.exports = fetchItem;

View File

@ -1,6 +1,8 @@
'use strict'; 'use strict';
const config = require('config'); const config = require('config');
const path = require('path');
const url = require('url');
const dateFns = require('date-fns'); const dateFns = require('date-fns');
const extensions = { const extensions = {
@ -10,45 +12,68 @@ const extensions = {
'video/webm': '.webm' 'video/webm': '.webm'
}; };
function interpolate(path, post, item) { function interpolate(pattern, user, post, item) {
const dateFormat = config.patterns.dateFormat || 'YYYYMMDD'; const dateFormat = config.library.dateFormat || 'YYYYMMDD';
const vars = {};
const vars = { if(config.library.base) {
pattern = path.join(config.library.base, pattern);
}
if(user) {
Object.assign(vars, {
$user: user.name,
$username: user.name,
$userId: user.id,
$userCreated: dateFns.format(user.created, dateFormat),
$userVerified: user.verified ? config.library.booleans.verified : '',
$userVerifiedEmail: user.verifiedEmail ? config.library.booleans.verifiedEmail : '',
$userGold: user.gold ? config.library.booleans.gold : '',
$profileId: user.profile.id,
$profileTitle: user.profile.title,
$profileDescription: user.profile.description,
$profileOver18: user.profile.over18 ? config.library.booleans.over18 : ''
});
}
if(post) {
Object.assign(vars, {
$postId: post.id, $postId: post.id,
$postTitle: (post.title || '').slice(0, config.patterns.titleLength), $postTitle: (post.title || '').slice(0, config.library.titleLength),
$postUser: post.user, $postUser: post.user || user.user,
$postDate: dateFns.format(post.datetime, dateFormat), $postDate: dateFns.format(post.datetime, dateFormat),
$postIndex: post.index + config.patterns.indexOffset, $postIndex: post.index + config.library.indexOffset,
$host: post.host.label $host: post.host.label
}; });
if(post.content.album) { if(post.content.album) {
Object.assign(vars, { Object.assign(vars, {
$albumId: post.content.album.id, $albumId: post.content.album.id,
$albumTitle: (post.content.album.title || '').slice(0, config.patterns.titleLength), $albumTitle: (post.content.album.title || '').slice(0, config.library.titleLength),
$albumDescription: post.content.album.description, $albumDescription: post.content.album.description,
$albumDate: dateFns.format(post.content.album.datetime, dateFormat) $albumDate: dateFns.format(post.content.album.datetime, dateFormat)
}); });
} }
}
if(item) { if(item) {
Object.assign(vars, { Object.assign(vars, {
$itemId: item.id, $itemId: item.id,
$itemTitle: (item.title || '').slice(0, config.patterns.titleLength), $itemTitle: (item.title || '').slice(0, config.library.titleLength),
$itemDescription: item.description, $itemDescription: item.description,
$itemDate: dateFns.format(item.datetime, dateFormat), $itemDate: dateFns.format(item.datetime, dateFormat),
$itemIndex: item.index + config.patterns.indexOffset, $itemIndex: item.index + config.library.indexOffset,
$extracted: item.extracted ? config.patterns.extractedLabel : '', $extracted: item.extracted ? config.library.booleans.extracted : '',
$ext: extensions[item.type] $ext: item.type ? extensions[item.type] : path.extname(url.parse(item.url).pathname)
}); });
} }
return Object.entries(vars).reduce((acc, [key, value], index) => { return Object.entries(vars).reduce((acc, [key, value], index) => {
// strip slashes for filesystem compatability // substitute slashes for filesystem compatability
value = (value || '').toString().replace(/\//g, config.patterns.slashSubstitute); value = (value || '').toString().replace(/\//g, config.library.slashSubstitute);
return acc.replace(key, value); return acc.replace(key, value);
}, path); }, pattern);
}; };
module.exports = interpolate; module.exports = interpolate;

19
save.js Normal file
View File

@ -0,0 +1,19 @@
'use strict';
const fs = require('fs-extra');
function save(filepath, stream) {
const file = fs.createWriteStream(filepath);
return new Promise((resolve, reject) => {
stream.pipe(file).on('error', error => {
reject(error);
}).on('finish', () => {
console.log('\x1b[32m%s\x1b[0m', `Saved '${filepath}'`);
resolve(filepath);
});
});
};
module.exports = save;

14
textToStream.js Normal file
View File

@ -0,0 +1,14 @@
'use strict';
const Readable = require('stream').Readable;
function textToStream(text) {
const stream = new Readable();
stream.push(text);
stream.push(null);
return stream;
};
module.exports = textToStream;