create upload/append-to-sheet script
This commit is contained in:
parent
014fbebb81
commit
9b29df1eab
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
node_modules
|
||||||
|
.credentials
|
168
index.js
Normal file
168
index.js
Normal file
@ -0,0 +1,168 @@
|
|||||||
|
var fs = require('fs');
|
||||||
|
var readline = require('readline');
|
||||||
|
var google = require('googleapis');
|
||||||
|
var googleAuth = require('google-auth-library');
|
||||||
|
var books = require('google-books-search');
|
||||||
|
var _ = require('lodash');
|
||||||
|
|
||||||
|
var SCOPES = ['https://www.googleapis.com/auth/spreadsheets'];
|
||||||
|
var TOKEN_DIR = '.credentials/';
|
||||||
|
var TOKEN_PATH = TOKEN_DIR + 'sheets.googleapis.com-my-library.json';
|
||||||
|
|
||||||
|
// Load client secrets from a local file.
|
||||||
|
fs.readFile(TOKEN_DIR + 'client_secret.json', function processClientSecrets(err, content) {
|
||||||
|
if (err) {
|
||||||
|
console.log('Error loading client secret file: ' + err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Authorize a client with the loaded credentials, then call the
|
||||||
|
// Google Sheets API.
|
||||||
|
authorize(JSON.parse(content), inputLoop);
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an OAuth2 client with the given credentials, and then execute the
|
||||||
|
* given callback function.
|
||||||
|
*
|
||||||
|
* @param {Object} credentials The authorization client credentials.
|
||||||
|
* @param {function} callback The callback to call with the authorized client.
|
||||||
|
*/
|
||||||
|
function authorize(credentials, callback) {
|
||||||
|
var clientSecret = credentials.installed.client_secret;
|
||||||
|
var clientId = credentials.installed.client_id;
|
||||||
|
var redirectUrl = credentials.installed.redirect_uris[0];
|
||||||
|
var auth = new googleAuth();
|
||||||
|
var oauth2Client = new auth.OAuth2(clientId, clientSecret, redirectUrl);
|
||||||
|
|
||||||
|
// Check if we have previously stored a token.
|
||||||
|
fs.readFile(TOKEN_PATH, function(err, token) {
|
||||||
|
if (err) {
|
||||||
|
getNewToken(oauth2Client, callback);
|
||||||
|
} else {
|
||||||
|
oauth2Client.credentials = JSON.parse(token);
|
||||||
|
callback(oauth2Client);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get and store new token after prompting for user authorization, and then
|
||||||
|
* execute the given callback with the authorized OAuth2 client.
|
||||||
|
*
|
||||||
|
* @param {google.auth.OAuth2} oauth2Client The OAuth2 client to get token for.
|
||||||
|
* @param {getEventsCallback} callback The callback to call with the authorized
|
||||||
|
* client.
|
||||||
|
*/
|
||||||
|
function getNewToken(oauth2Client, callback) {
|
||||||
|
var authUrl = oauth2Client.generateAuthUrl({
|
||||||
|
access_type: 'offline',
|
||||||
|
scope: SCOPES
|
||||||
|
});
|
||||||
|
console.log('Authorize this app by visiting this url: ', authUrl);
|
||||||
|
var rl = readline.createInterface({
|
||||||
|
input: process.stdin,
|
||||||
|
output: process.stdout
|
||||||
|
});
|
||||||
|
rl.question('Enter the code from that page here: ', function(code) {
|
||||||
|
rl.close();
|
||||||
|
oauth2Client.getToken(code, function(err, token) {
|
||||||
|
if (err) {
|
||||||
|
console.log('Error while trying to retrieve access token', err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
oauth2Client.credentials = token;
|
||||||
|
storeToken(token);
|
||||||
|
callback(oauth2Client);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Store token to disk be used in later program executions.
|
||||||
|
*
|
||||||
|
* @param {Object} token The token to store to disk.
|
||||||
|
*/
|
||||||
|
function storeToken(token) {
|
||||||
|
try {
|
||||||
|
fs.mkdirSync(TOKEN_DIR);
|
||||||
|
} catch (err) {
|
||||||
|
if (err.code != 'EEXIST') {
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fs.writeFile(TOKEN_PATH, JSON.stringify(token));
|
||||||
|
console.log('Token stored to ' + TOKEN_PATH);
|
||||||
|
}
|
||||||
|
|
||||||
|
function inputLoop(auth) {
|
||||||
|
insertRow(auth);
|
||||||
|
}
|
||||||
|
|
||||||
|
function insertRow(auth) {
|
||||||
|
var rl = readline.createInterface({
|
||||||
|
input: process.stdin,
|
||||||
|
output: process.stdout
|
||||||
|
});
|
||||||
|
rl.question('ISBN: ', function(isbn) {
|
||||||
|
rl.close();
|
||||||
|
books.search(isbn, {
|
||||||
|
field: 'isbn',
|
||||||
|
lang: 'en'
|
||||||
|
}, function(error, results) {
|
||||||
|
if ( ! error && results.length > 0) {
|
||||||
|
var book = normalizeGoogleData(results[0]);
|
||||||
|
console.log(book);
|
||||||
|
appendToSheet(auth, _.values(book));
|
||||||
|
} else if ( results.length == 0) {
|
||||||
|
console.log("no book found");
|
||||||
|
inputLoop(auth);
|
||||||
|
} else {
|
||||||
|
console.log(error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeGoogleData(book) {
|
||||||
|
var normalized = {
|
||||||
|
title: book.subtitle ? book.title + ': ' + book.subtitle : book.title,
|
||||||
|
author: _.join(book.authors, ', '),
|
||||||
|
authorLast: _.lowerCase(_.reverse(_.split(book.authors[0], ' '))[0]),
|
||||||
|
"isbn-10": _.find(book.industryIdentifiers, { type: "ISBN_10" }).identifier || '',
|
||||||
|
"isbn-13": _.find(book.industryIdentifiers, { type: "ISBN_13" }).identifier || '',
|
||||||
|
format: '',
|
||||||
|
genre: book.categories[0],
|
||||||
|
publisher: _.trim(book.publisher, '"'),
|
||||||
|
series: '',
|
||||||
|
volume: '',
|
||||||
|
publishedDate: book.publishedDate.substring(0, 4),
|
||||||
|
coverurl: '',
|
||||||
|
description: book.description,
|
||||||
|
notes: '',
|
||||||
|
signed: ''
|
||||||
|
};
|
||||||
|
return normalized;
|
||||||
|
}
|
||||||
|
|
||||||
|
function appendToSheet(auth, book) {
|
||||||
|
var sheets = google.sheets('v4');
|
||||||
|
sheets.spreadsheets.values.append({
|
||||||
|
auth: auth,
|
||||||
|
spreadsheetId: '1w5Dc57wV0_rrKFsG7KM-qdPWEpqYk6lFu3JzAA0cSv0',
|
||||||
|
range: 'Sheet1',
|
||||||
|
valueInputOption: "RAW",
|
||||||
|
insertDataOption: "INSERT_ROWS",
|
||||||
|
resource: {
|
||||||
|
"range": "Sheet1",
|
||||||
|
"majorDimension": "ROWS",
|
||||||
|
"values": [
|
||||||
|
book
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}, function(err, response) {
|
||||||
|
if (err) {
|
||||||
|
console.log('The API returned an error: ' + err);
|
||||||
|
}
|
||||||
|
inputLoop(auth);
|
||||||
|
});
|
||||||
|
}
|
25
package.json
Normal file
25
package.json
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"name": "library",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "Managing my personal library",
|
||||||
|
"main": "index.js",
|
||||||
|
"scripts": {
|
||||||
|
"test": "echo \"Error: no test specified\" && exit 1"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/deltamualpha/my-library.git"
|
||||||
|
},
|
||||||
|
"author": "https://github.com/deltamualpha/",
|
||||||
|
"license": "BSD-3-Clause",
|
||||||
|
"bugs": {
|
||||||
|
"url": "https://github.com/deltamualpha/my-library/issues"
|
||||||
|
},
|
||||||
|
"homepage": "https://github.com/deltamualpha/my-library",
|
||||||
|
"dependencies": {
|
||||||
|
"google-auth-library": "^0.10.0",
|
||||||
|
"google-books-search": "^0.3.1",
|
||||||
|
"googleapis": "^19.0.0",
|
||||||
|
"lodash": "^4.17.4"
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user