/**
* @module wolfram-alpha-api
*/
const https = require('https');
const querystring = require('querystring');
const baseApiUrl = 'https://api.wolframalpha.com/';
const createApiParamsRejectMsg = 'method only receives string or object';
/**
* We support four 'output' formats:
* 'string' and 'xml' are both strings,
* 'json' is an Object (a result of JSON.parse), and
* 'image' is a string of a "Data URI"
* @typedef OutputFormat
* @property {'string'|'json'|'image'|'xml'} output
*/
/**
* @typedef FetchParams
* @property {Object} params
* @property {string} params.url - full URL of api call
* @property {OutputFormat} params.output - which OutputFormat do we want?
* @example {url: 'https://api.wolframalpha.com/v1/result?appid=DEMO&i=2%2B2', output: 'string'}
*/
/**
* @typedef FormatParams
* @property {Object} params
* @property {string} params.data - data returned by fetchResults
* @property {OutputFormat} params.output - which OutputFormat do we want?
* @property {integer} statusCode - HTTP status code of fetchResults
* @property {string} contentType - HTTP content-type header from fetchResults
* @example {data: '4', output: 'string', statusCode: 200, contentType: 'text/plain;charset=utf-8'}
*/
/**
* Build a URL call from a baseUrl and input; specify an OutputFormat (for fetchResults).
* @param {string} baseUrl - base URL of API we are trying to call
* @param {(string|Object)} input - string of query, or object of parameters
* @param {OutputFormat} [output=string] - which OutputFormat we want
* @returns {Promise<FetchParams>}
* @example
* // resolves {url: 'https://api.wolframalpha.com/v1/result?appid=DEMO&i=2%2B2', output: 'string'}
* createApiParams('https://api.wolframalpha.com/v1/result?appid=DEMO', '2+2', 'string')
* // resolves {
* // url: 'https://api.wolframalpha.com/v1/simple?appid=DEMO&i=nyc%20to%la&units=metric',
* // output: 'image'
* // }
* createApiParams('https://api.wolframalpha.com/v1/simple?appid=DEMO',
* {i: 'nyc to la', units: 'metric'}, 'image')
* // rejects TypeError('method only receives string or object')
* createApiParams('https://api.wolframalpha.com/v1/result?appid=DEMO')
*/
function createApiParams(baseUrl, input, output = 'string') {
return new Promise((resolve, reject) => {
switch (typeof input) {
case 'string':
resolve({ url: `${baseUrl}&i=${encodeURIComponent(input)}`, output });
break;
case 'object':
resolve({ url: `${baseUrl}&${querystring.stringify(input)}`, output });
break;
default:
reject(new TypeError(createApiParamsRejectMsg));
}
});
}
/**
* Return a Promise that downloads params.url, and resolves the results (for formatResults).
* @param {FetchParams} params
* @returns {Promise<FormatParams>}
* @example
* // resolves { data: '4', output: 'string', statusCode: 200,
* // contentType: 'text/plain;charset=utf-8' }
* fetchResults({
* url: 'https://api.wolframalpha.com/v1/result?appid=DEMO&i=2%2B2',
* output: 'string'
* })
* // resolves { output: 'json', statusCode: 200, contentType: 'text/plain;charset=utf-8',
* // data: '{"queryresult" : {\n\t"success" : true, \n\t"error" : false, \n\t"nu...', }
* fetchResults({
* url: 'https://api.wolframalpha.com/v2/query?appid=DEMO&input=2%2B2&output=json',
* output: 'json'
* })
* // resolves { output: 'image', statusCode: 200, contentType: 'image/gif',
* // data: 'R0lGODlhHAJNBfcAAAAAAAAEAAgICAgMCBAQEBAUEBgYGBgcGCAgICAkICksKSkoKTEwMT...'}
* fetchResults({
* url: 'https://api.wolframalpha.com/v1/simple?appid=DEMO&i=nyc%20to%la&units=metric',
* output: 'image'
* })
* // resolves { output: 'image', statusCode: 501, contentType: 'text/plain;charset=utf-8',
* // data: 'Wolfram|Alpha did not understand your input' }
* fetchResults({
* url: 'https://api.wolframalpha.com/v1/result?appid=DEMO&i=F9TVlu5AmVzL'
* output: 'string'
* })
*/
function fetchResults(params) {
const { url, output } = params;
return new Promise((resolve, reject) => {
https
.get(url, (res) => {
const statusCode = res.statusCode;
const contentType = res.headers['content-type'];
if (output === 'image' && statusCode === 200) {
res.setEncoding('base64'); // API returns binary data, we want base64 for the Data URI
} else {
res.setEncoding('utf8');
}
let data = '';
res.on('data', (chunk) => {
data += chunk;
});
res.on('end', () => {
resolve({ data, output, statusCode, contentType });
});
})
.on('error', (e) => {
reject(e);
});
});
}
/**
* Return a Promise that resolves a formatted form of params.data, as specified by
* params.output, params.statusCode, and params.contentType
* @param {FormatParams} params
* @returns {Promise<(Object|string)>}
* @example
* // resolves "4"
* formatResults({
* data: '4', output: 'string', statusCode: 200,
* contentType: 'text/plain;charset=utf-8'
* })
* // resolves {success: true, error: false, numpods: 6, datatypes: 'Math', timedout: '' ...}
* formatResults({
* data: '{"queryresult" : {\n\t"success" : true, \n\t"error" : false, \n\t"nu...',
* output: 'json', statusCode: 200, contentType: 'text/plain;charset=utf-8'
* })
* // resolves 'data:image/gif;base64,R0lGODlhHAJNBfcAAAAAAAAEAAgICAgMCBAQEBAUEBgYGBgcGCAgICAkIC...
* formatResults({
* data: 'R0lGODlhHAJNBfcAAAAAAAAEAAgICAgMCBAQEBAUEBgYGBgcGCAgICAkICksKSkoKTEwMT...'
* output: 'image', statusCode: 200, contentType: 'image/gif'
* })
* // rejects Error('Wolfram|Alpha did not understand your input')
* formatResults({
* data: 'Wolfram|Alpha did not understand your input'
* output: 'image', statusCode: 501, contentType: 'text/plain;charset=utf-8'
* })
*/
function formatResults(params) {
const { data, output, statusCode, contentType } = params;
return new Promise((resolve, reject) => {
if (statusCode === 200) {
switch (output) {
case 'json':
try {
resolve(JSON.parse(data).queryresult);
} catch (e) {
reject(
new Error('Temporary problem in parsing JSON, please try again.'),
);
}
break;
case 'image':
resolve(`data:${contentType};base64,${data}`);
break;
default:
resolve(data);
}
// if (statusCode !== 200)...
} else if (/^text\/html/.test(contentType)) {
// Rarely, there may be a catastrophic error where the API gives an HTML error page.
reject(new Error('Temporary problem with the API, please try again.'));
} else {
// This runs if non-full API input is empty, ambiguous, or otherwise invalid.
reject(new Error(data));
}
});
}
/**
* Wolfram|Alpha API NPM Library
*/
class WolframAlphaAPI {
/**
* You may get your 'appid' at {@link https://developer.wolframalpha.com/portal/myapps/}.
* Remember, this appID must be kept secret.
* @param {string} appid - the appid, must be non-empty string.
* @throws TypeError
* @example
* const WolframAlphaAPI = require('wolfram-alpha-api');
* const waApi = WolframAlphaAPI('DEMO-APPID');
*/
constructor(appid) {
if (!appid || typeof appid !== 'string') {
throw new TypeError('appid must be non-empty string');
}
this.appid = appid;
}
/**
* Takes 'input' (which is either a string, or an object of parameters), runs it through
* the Wolfram|Alpha Simple API, and returns a Promise that
* resolves a string of a "Data URI", or rejects if there's an error.
* @param {string|Object} input - string or object of parameters
* @returns {Promise<DataURI>}
* @see https://products.wolframalpha.com/simple-api/documentation/
* @example
* // "data:image/gif;base64,R0lGODlhHAK5AvcAAAAAAAAEAAgICAgMCBAQEBAUEBsdGzE0MTk8OTk4OS0uLSAkI...
* waApi.getSimple('2+2').then(console.log, console.error);
* // "data:image/gif;base64,R0lGODlhHAJNBfcAAAAAAAAEAAgICAgMCBAQEBAUEBgYGBgcGCAgICAkICksKSkoK...
* waApi.getSimple({i: 'nyc to la', units: 'metric'}).then(console.log, console.error);
* // Error: Wolfram|Alpha did not understand your input
* waApi.getSimple('F9TVlu5AmVzL').then(console.log, console.error);
* // TypeError: method only receives string or object
* waApi.getSimple().then(console.log, console.error);
*/
getSimple(input) {
const baseUrl = `${baseApiUrl}v1/simple?appid=${this.appid}`;
return createApiParams(baseUrl, input, 'image')
.then(fetchResults)
.then(formatResults);
}
/**
* Takes 'input' (which is either a string, or an object of parameters), runs it through
* the Wolfram|Alpha Short Answers API, and returns a Promise that
* resolves a string of results, or rejects if there's an error.
* @param {string|Object} input - string or object of parameters
* @returns {Promise<string>}
* @see https://products.wolframalpha.com/short-answers-api/documentation/
* @example
* // "4"
* waApi.getShort('2+2').then(console.log, console.error);
* // "3966 kilometers"
* waApi.getShort({i: 'nyc to la', units: 'metric'}).then(console.log, console.error);
* // Error: Wolfram|Alpha did not understand your input
* waApi.getShort('F9TVlu5AmVzL').then(console.log, console.error);
* // TypeError: method only receives string or object
* waApi.getShort().then(console.log, console.error);
*/
getShort(input) {
const baseUrl = `${baseApiUrl}v1/result?appid=${this.appid}`;
return createApiParams(baseUrl, input)
.then(fetchResults)
.then(formatResults);
}
/**
* Takes 'input' (which is either a string, or an object of parameters), runs it through
* the Wolfram|Alpha Spoken Results API, and returns a Promise that
* resolves a string of results, or rejects if there's an error.
* @param {string|Object} input - string or object of parameters
* @returns {Promise<string>}
* @see https://products.wolframalpha.com/spoken-results-api/documentation/
* @example
* // "The answer is 4"
* waApi.getSpoken('2+2').then(console.log, console.error);
* // "The answer is about 3966 kilometers"
* waApi.getSpoken({i: 'nyc to la', units: 'metric'}).then(console.log, console.error);
* // Error: Wolfram Alpha did not understand your input
* waApi.getSpoken('F9TVlu5AmVzL').then(console.log, console.error);
* // TypeError: method only receives string or object
* waApi.getSpoken().then(console.log, console.error);
*/
getSpoken(input) {
const baseUrl = `${baseApiUrl}v1/spoken?appid=${this.appid}`;
return createApiParams(baseUrl, input)
.then(fetchResults)
.then(formatResults);
}
/**
* Takes 'input' (which is either a string, or an object of parameters), runs it through
* the Wolfram|Alpha Full Results API, and returns a Promise that
* either resolves an Object or a string of XML, or rejects if there's an error.
* @param {string|Object} input - string or object of parameters
* @returns {Promise<(Object|string)>}
* @see https://products.wolframalpha.com/api/documentation/
* @example
* // {success: true, error: false, numpods: 6, datatypes: 'Math', timedout: '', timing: 1.08 ...
* waApi.getFull('2+2').then(console.log, console.error);
* // "<queryresult success='true' error='false' numpods='7' ...
* waApi.getFull({input:'nyc to la', output:'xml'}).then(console.log, console.error);
* // { success: false, error: false, numpods: 0, datatypes: '', timedout: '', ...
* waApi.getFull('F9TVlu5AmVzL').then(console.log, console.error)
* // TypeError: method only receives string or object
* waApi.getFull().then(console.log, console.error);
*/
getFull(input) {
const baseUrl = `${baseApiUrl}v2/query?appid=${this.appid}`;
// This promise works just like createApiParams, except with a bit more processing
return new Promise((resolve, reject) => {
switch (typeof input) {
case 'string':
resolve({
url: `${baseUrl}&input=${encodeURIComponent(input)}&output=json`,
output: 'json',
});
break;
case 'object': {
// the API defaults to XML, but we want to default to JSON.
const options = Object.assign({ output: 'json' }, input);
// since all other APIs use 'i' instead of 'input', allow for 'i'.
if (options.input == null && options.i != null) {
options.input = options.i;
delete options.i;
}
resolve({
url: `${baseUrl}&${querystring.stringify(options)}`,
output: options.output,
});
break;
}
default:
reject(new TypeError(createApiParamsRejectMsg));
}
})
.then(fetchResults)
.then(formatResults);
}
}
/**
* You may get your 'appid' at {@link https://developer.wolframalpha.com/portal/myapps/}.
* Remember, this appID must be kept secret.
* @param {string} appid - the appid, must be non-empty string.
* @throws TypeError
* @example
* const WolframAlphaAPI = require('wolfram-alpha-api');
* const waApi = WolframAlphaAPI('DEMO-APPID');
*/
function initializeClass(appid) {
return new WolframAlphaAPI(appid);
}
module.exports = initializeClass;