@meteor-it/xrest
Advanced tools
Comparing version 0.2.1 to 0.3.1
249
index.js
@@ -1,21 +0,28 @@ | ||
import Logger from '@meteor-it/logger'; | ||
import * as multipart from './multipart'; | ||
import {EventEmitter} from 'events'; | ||
import {METHODS} from 'http'; | ||
import * as http from 'http'; | ||
import * as https from 'https'; | ||
import {parse as parseUrl,resolve} from 'url'; | ||
import {stringify} from 'querystring'; | ||
import * as zlib from 'zlib'; | ||
import iconv from 'iconv-lite'; | ||
export * from './multipart'; | ||
const POSSIBLE_EVENTS=[...METHODS]; | ||
const POSSIBLE_MIDDLEWARES=['STREAM']; | ||
"use strict"; | ||
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { | ||
return new (P || (P = Promise))(function (resolve, reject) { | ||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } | ||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } | ||
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); } | ||
step((generator = generator.apply(thisArg, _arguments || [])).next()); | ||
}); | ||
}; | ||
function __export(m) { | ||
for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p]; | ||
} | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
const logger_1 = require("@meteor-it/logger"); | ||
const multipart = require("./multipart"); | ||
const events_1 = require("events"); | ||
const http_1 = require("http"); | ||
const http = require("http"); | ||
const https = require("https"); | ||
const url_1 = require("url"); | ||
const querystring_1 = require("querystring"); | ||
const zlib = require("zlib"); | ||
const iconv_lite_1 = require("iconv-lite"); | ||
__export(require("./multipart")); | ||
const POSSIBLE_EVENTS = [...http_1.METHODS]; | ||
const POSSIBLE_MIDDLEWARES = ['STREAM']; | ||
const USER_AGENT = 'Meteor-IT XRest'; | ||
const decoders = { | ||
@@ -29,59 +36,47 @@ gzip(buf, callback) { | ||
}; | ||
const parsers={ | ||
json(text,cb){ | ||
try{ | ||
cb(null,JSON.parse(text)); | ||
}catch(e){ | ||
cb(null,text); | ||
const parsers = { | ||
json(text, cb) { | ||
try { | ||
cb(null, JSON.parse(text)); | ||
} | ||
catch (e) { | ||
cb(null, text); | ||
} | ||
} | ||
}; | ||
class Request extends EventEmitter { | ||
url; | ||
options; | ||
headers; | ||
class Request extends events_1.EventEmitter { | ||
constructor(url, options) { | ||
super(); | ||
this.prepare(url,options); | ||
this.prepare(url, options); | ||
} | ||
prepare(url,options){ | ||
logger.debug('prepare(%s)',url); | ||
if(url.indexOf('undefined')+1){ | ||
prepare(url, options) { | ||
logger.debug('prepare(%s)', url); | ||
if (url.indexOf('undefined') + 1) { | ||
logger.warn('undefined found in request url! Stack for reference:'); | ||
logger.warn(new Error('reference stack').stack); | ||
} | ||
this.url = parseUrl(url); | ||
if(!this.url.hostname) | ||
this.url = url_1.parse(url); | ||
if (!this.url.hostname) | ||
console.log(this.url); | ||
this.options = options; | ||
this.headers = { | ||
'Accept': '*/*', | ||
'User-Agent': USER_AGENT, | ||
'Host': this.url.host, | ||
'Accept-Encoding': 'gzip, deflate', | ||
...options.headers | ||
}; | ||
this.headers = Object.assign({ 'Accept': '*/*', 'User-Agent': USER_AGENT, 'Host': this.url.host, 'Accept-Encoding': 'gzip, deflate' }, options.headers); | ||
// set port and method defaults | ||
if (!this.url.port) | ||
if (!this.url.port) | ||
this.url.port = (this.url.protocol == 'https:') ? '443' : '80'; | ||
if (!this.options.method) | ||
if (!this.options.method) | ||
this.options.method = (this.options.data) ? 'POST' : 'GET'; | ||
if (typeof this.options.followRedirects == 'undefined') | ||
if (typeof this.options.followRedirects == 'undefined') | ||
this.options.followRedirects = true; | ||
if(this.options.timeout===undefined) | ||
this.options.timeout=12000; | ||
if(!this.options.parser) | ||
this.options.parser=parsers.json; | ||
if (this.options.timeout === undefined) | ||
this.options.timeout = 12000; | ||
if (!this.options.parser) | ||
this.options.parser = parsers.json; | ||
// stringify query given in options of not given in URL | ||
if (this.options.query) { | ||
if (typeof this.options.query == 'object') | ||
this.url.query = stringify(this.options.query); | ||
else this.url.query = this.options.query; | ||
this.url.query = querystring_1.stringify(this.options.query); | ||
else | ||
this.url.query = this.options.query; | ||
} | ||
this.applyAuth(); | ||
if (this.options.multipart) { | ||
@@ -97,3 +92,3 @@ this.headers['Content-Type'] = `multipart/form-data; boundary=${multipart.DEFAULT_BOUNDARY}`; | ||
if (typeof this.options.data == 'object' && !Buffer.isBuffer(this.options.data)) { | ||
this.options.data = stringify(this.options.data); | ||
this.options.data = querystring_1.stringify(this.options.data); | ||
this.headers['Content-Type'] = 'application/x-www-form-urlencoded'; | ||
@@ -111,5 +106,3 @@ this.headers['Content-Length'] = this.options.data.length; | ||
} | ||
const proto = (this.url.protocol == 'https:') ? https : http; | ||
this.request = proto.request({ | ||
@@ -124,3 +117,2 @@ host: this.url.hostname, | ||
}); | ||
this.makeRequest(); | ||
@@ -133,4 +125,6 @@ } | ||
let path = this.url.pathname || '/'; | ||
if (this.url.hash) path += this.url.hash; | ||
if (this.url.query) path += `?${this.url.query}`; | ||
if (this.url.hash) | ||
path += this.url.hash; | ||
if (this.url.query) | ||
path += `?${this.url.query}`; | ||
return path; | ||
@@ -140,3 +134,2 @@ } | ||
let authParts; | ||
if (this.url.auth) { | ||
@@ -147,3 +140,2 @@ authParts = this.url.auth.split(':'); | ||
} | ||
if (this.options.username && this.options.password !== undefined) { | ||
@@ -163,3 +155,3 @@ const b = new Buffer([this.options.username, this.options.password].join(':')); | ||
if (response.statusCode === 303) { | ||
this.url = parseUrl(resolve(this.url.href, response.headers['location'])); | ||
this.url = url_1.parse(url_1.resolve(this.url.href, response.headers['location'])); | ||
this.options.method = 'GET'; | ||
@@ -170,3 +162,3 @@ delete this.options.data; | ||
else { | ||
this.url = parseUrl(resolve(this.url.href, response.headers['location'])); | ||
this.url = url_1.parse(url_1.resolve(this.url.href, response.headers['location'])); | ||
this.reRetry(); | ||
@@ -183,11 +175,8 @@ // TODO: Handle somehow infinite redirects | ||
let body = ''; | ||
// When using browserify, response.setEncoding is not defined | ||
if (typeof response.setEncoding == 'function') | ||
response.setEncoding('binary'); | ||
response.on('data', chunk => { | ||
body += chunk; | ||
}); | ||
response.on('end', () => { | ||
@@ -216,3 +205,2 @@ response.rawEncoded = body; | ||
const decoder = response.headers['content-encoding']; | ||
if (decoder in decoders) { | ||
@@ -233,5 +221,5 @@ decoders[decoder].call(response, body, callback); | ||
try { | ||
return iconv.decode(body, charset); | ||
return iconv_lite_1.default.decode(body, charset); | ||
} | ||
catch (err) {} | ||
catch (err) { } | ||
} | ||
@@ -273,3 +261,3 @@ } | ||
fireSuccess(body, response) { | ||
if (parseInt(response.statusCode,10) >= 400) { | ||
if (parseInt(response.statusCode, 10) >= 400) { | ||
this.emit('fail', body, response); | ||
@@ -303,3 +291,3 @@ } | ||
reRetry() { | ||
this.request.removeAllListeners().on('error', () => {}); | ||
this.request.removeAllListeners().on('error', () => { }); | ||
if (this.request.finished) { | ||
@@ -311,16 +299,17 @@ this.request.abort(); | ||
} | ||
async run() { | ||
if (this.options.multipart) { | ||
await multipart.write(this.request, this.options.data, () => { | ||
run() { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
if (this.options.multipart) { | ||
yield multipart.write(this.request, this.options.data, () => { | ||
this.request.end(); | ||
}); | ||
} | ||
else { | ||
if (this.options.data) { | ||
this.request.write(this.options.data, this.options.encoding || 'utf8'); | ||
} | ||
this.request.end(); | ||
}); | ||
} | ||
else { | ||
if (this.options.data) { | ||
this.request.write(this.options.data, this.options.encoding || 'utf8'); | ||
} | ||
this.request.end(); | ||
} | ||
return this; | ||
return this; | ||
}); | ||
} | ||
@@ -340,3 +329,2 @@ abort(err) { | ||
} | ||
this.request.on('close', () => { | ||
@@ -350,3 +338,2 @@ if (err) { | ||
}); | ||
this.aborted = true; | ||
@@ -358,3 +345,3 @@ this.request.abort(); | ||
retry(timeout) { | ||
timeout = parseInt(timeout,10); | ||
timeout = parseInt(timeout, 10); | ||
const fn = this.reRetry.bind(this); | ||
@@ -370,49 +357,48 @@ if (!isFinite(timeout) || timeout <= 0) { | ||
} | ||
const logger=new Logger('xrest'); | ||
export function emit(eventString,options={}) { | ||
let [event,path,...middlewares]=eventString.split(' '); | ||
let middleFunctions=[]; | ||
for(let middleware of middlewares){ | ||
if(middleware.toUpperCase()!==middleware){ | ||
logger.warn('Upper case is preffered for middleware names! (Got: %s)',event); | ||
middleware=middleware.toUpperCase(); | ||
const logger = new logger_1.default('xrest'); | ||
function emit(eventString, options = {}) { | ||
let [event, path, ...middlewares] = eventString.split(' '); | ||
let middleFunctions = []; | ||
for (let middleware of middlewares) { | ||
if (middleware.toUpperCase() !== middleware) { | ||
logger.warn('Upper case is preffered for middleware names! (Got: %s)', event); | ||
middleware = middleware.toUpperCase(); | ||
} | ||
if(!~POSSIBLE_MIDDLEWARES.indexOf(middleware)) | ||
throw new Error('Unknown middleware: '+middleware); | ||
if (!~POSSIBLE_MIDDLEWARES.indexOf(middleware)) | ||
throw new Error('Unknown middleware: ' + middleware); | ||
middleFunctions.push(middleware); | ||
} | ||
if(event.toUpperCase()!==event){ | ||
logger.warn('Upper case is preffered for event names! (Got: %s)',event); | ||
event=event.toUpperCase(); | ||
if (event.toUpperCase() !== event) { | ||
logger.warn('Upper case is preffered for event names! (Got: %s)', event); | ||
event = event.toUpperCase(); | ||
} | ||
if(!~POSSIBLE_EVENTS.indexOf(event)){ | ||
throw new Error('Unknown event: '+event+', possible events are '+POSSIBLE_EVENTS.join(', ')+'!'); | ||
if (!~POSSIBLE_EVENTS.indexOf(event)) { | ||
throw new Error('Unknown event: ' + event + ', possible events are ' + POSSIBLE_EVENTS.join(', ') + '!'); | ||
} | ||
options.method = event; | ||
const request=new Request(path,options); | ||
if(~middleFunctions.indexOf('STREAM')){ | ||
const request = new Request(path, options); | ||
if (~middleFunctions.indexOf('STREAM')) { | ||
logger.debug('Streaming'); | ||
return new Promise((res,rej)=>{ | ||
return new Promise((res, rej) => { | ||
request.run(); | ||
request.on('timeout',(ms)=>{ | ||
rej(new Error('Timeout: '+ms)); | ||
request.on('timeout', (ms) => { | ||
rej(new Error('Timeout: ' + ms)); | ||
}); | ||
request.on('response',(response)=>{ | ||
request.on('response', (response) => { | ||
res(response); | ||
}); | ||
}); | ||
}else{ | ||
return new Promise((res,rej)=>{ | ||
} | ||
else { | ||
return new Promise((res, rej) => { | ||
request.run(); | ||
request.on('timeout',(ms)=>{ | ||
rej(new Error('Timeout: '+ms)); | ||
request.on('timeout', (ms) => { | ||
rej(new Error('Timeout: ' + ms)); | ||
}); | ||
request.on('complete',(result,response)=>{ | ||
if(result instanceof Error) { | ||
request.on('complete', (result, response) => { | ||
if (result instanceof Error) { | ||
rej(result); | ||
return; | ||
} | ||
response.body=result; | ||
response.body = result; | ||
res(response); | ||
@@ -423,15 +409,16 @@ }); | ||
} | ||
export default class XRest{ | ||
baseUrl; | ||
defaultOptions; | ||
constructor(url,defaultOptions){ | ||
logger.debug('new XRest(%s)',url); | ||
this.baseUrl=url; | ||
this.defaultOptions=defaultOptions; | ||
exports.emit = emit; | ||
class XRest { | ||
constructor(url, defaultOptions) { | ||
logger.debug('new XRest(%s)', url); | ||
this.baseUrl = url; | ||
this.defaultOptions = defaultOptions; | ||
} | ||
emit(eventString,options){ | ||
let [event,path,...middlewares]=eventString.split(' '); | ||
path=resolve(this.baseUrl, path); | ||
return emit([event,path,...middlewares].join(' '),options); | ||
emit(eventString, options) { | ||
let [event, path, ...middlewares] = eventString.split(' '); | ||
path = url_1.resolve(this.baseUrl, path); | ||
return emit([event, path, ...middlewares].join(' '), options); | ||
} | ||
} | ||
} | ||
exports.default = XRest; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["xrest/index.js"],"names":[],"mappings":"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","file":"xrest/index.js","sourcesContent":["import Logger from '@meteor-it/logger';\n\nimport * as multipart from './multipart';\n\nimport {EventEmitter} from 'events';\nimport {METHODS} from 'http';\nimport * as http from 'http';\nimport * as https from 'https';\nimport {parse as parseUrl,resolve} from 'url';\nimport {stringify} from 'querystring';\nimport * as zlib from 'zlib';\nimport iconv from 'iconv-lite';\n\nexport * from './multipart';\n\nconst POSSIBLE_EVENTS=[...METHODS];\nconst POSSIBLE_MIDDLEWARES=['STREAM'];\n\nconst USER_AGENT = 'Meteor-IT XRest';\n\nconst decoders = {\n    gzip(buf, callback) {\n        zlib.gunzip(buf, callback);\n    },\n    deflate(buf, callback) {\n        zlib.inflate(buf, callback);\n    }\n};\nconst parsers={\n    json(text,cb){\n        try{\n            cb(null,JSON.parse(text));\n        }catch(e){\n            cb(null,text);\n        }\n    }\n};\n\nclass Request extends EventEmitter {\n    url;\n    options;\n    headers;\n    \n    constructor(url, options) {\n        super();\n        this.prepare(url,options);\n    }\n    prepare(url,options){\n        logger.debug('prepare(%s)',url);\n        if(url.indexOf('undefined')+1){\n            logger.warn('undefined found in request url! Stack for reference:');\n            logger.warn(new Error('reference stack').stack);\n        }\n        this.url = parseUrl(url);\n        if(!this.url.hostname)\n            console.log(this.url);\n        this.options = options;\n        this.headers = {\n            'Accept': '*/*',\n            'User-Agent': USER_AGENT,\n            'Host': this.url.host,\n            'Accept-Encoding': 'gzip, deflate',\n            ...options.headers\n        };\n\n        // set port and method defaults\n        if (!this.url.port) \n            this.url.port = (this.url.protocol == 'https:') ? '443' : '80';\n        if (!this.options.method) \n            this.options.method = (this.options.data) ? 'POST' : 'GET';\n        if (typeof this.options.followRedirects == 'undefined') \n            this.options.followRedirects = true;\n        if(this.options.timeout===undefined)\n            this.options.timeout=12000;\n        if(!this.options.parser)\n            this.options.parser=parsers.json;\n\n        // stringify query given in options of not given in URL\n        if (this.options.query) {\n            if (typeof this.options.query == 'object')\n                this.url.query = stringify(this.options.query);\n            else this.url.query = this.options.query;\n        }\n        this.applyAuth();\n\n        if (this.options.multipart) {\n            this.headers['Content-Type'] = `multipart/form-data; boundary=${multipart.DEFAULT_BOUNDARY}`;\n            const multipartSize = multipart.sizeOf(this.options.data, multipart.DEFAULT_BOUNDARY);\n            if (!isNaN(multipartSize))\n                this.headers['Content-Length'] = multipartSize;\n            else\n                throw new Error('Cannot get Content-Length!');\n        }\n        else {\n            if (typeof this.options.data == 'object' && !Buffer.isBuffer(this.options.data)) {\n                this.options.data = stringify(this.options.data);\n                this.headers['Content-Type'] = 'application/x-www-form-urlencoded';\n                this.headers['Content-Length'] = this.options.data.length;\n            }\n            if (typeof this.options.data == 'string') {\n                const buffer = new Buffer(this.options.data, this.options.encoding || 'utf8');\n                this.options.data = buffer;\n                this.headers['Content-Length'] = buffer.length;\n            }\n            if (!this.options.data) {\n                this.headers['Content-Length'] = 0;\n            }\n        }\n\n        const proto = (this.url.protocol == 'https:') ? https : http;\n\n        this.request = proto.request({\n            host: this.url.hostname,\n            port: this.url.port,\n            path: this.fullPath(),\n            method: this.options.method,\n            headers: this.headers,\n            rejectUnauthorized: this.options.rejectUnauthorized,\n            agent: this.options.agent\n        });\n\n        this.makeRequest();\n    }\n    isRedirect(response) {\n        return ([301, 302, 303, 307].includes(response.statusCode));\n    }\n    fullPath() {\n        let path = this.url.pathname || '/';\n        if (this.url.hash) path += this.url.hash;\n        if (this.url.query) path += `?${this.url.query}`;\n        return path;\n    }\n    applyAuth() {\n        let authParts;\n\n        if (this.url.auth) {\n            authParts = this.url.auth.split(':');\n            this.options.username = authParts[0];\n            this.options.password = authParts[1];\n        }\n\n        if (this.options.username && this.options.password !== undefined) {\n            const b = new Buffer([this.options.username, this.options.password].join(':'));\n            this.headers['Authorization'] = `Basic ${b.toString('base64')}`;\n        }\n        else if (this.options.accessToken) {\n            this.headers['Authorization'] = `Bearer ${this.options.accessToken}`;\n        }\n    }\n    responseHandler(response) {\n        if (this.isRedirect(response) && this.options.followRedirects) {\n            try {\n                // 303 should redirect and retrieve content with the GET method\n                // http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.3.4\n                if (response.statusCode === 303) {\n                    this.url = parseUrl(resolve(this.url.href, response.headers['location']));\n                    this.options.method = 'GET';\n                    delete this.options.data;\n                    this.reRetry();\n                }\n                else {\n                    this.url = parseUrl(resolve(this.url.href, response.headers['location']));\n                    this.reRetry();\n                    // TODO: Handle somehow infinite redirects\n                }\n            }\n            catch (err) {\n                err.message = `Failed to follow redirect: ${err.message}`;\n                this.fireError(err, response);\n            }\n        }\n        else {\n            let body = '';\n\n            // When using browserify, response.setEncoding is not defined\n            if (typeof response.setEncoding == 'function')\n                response.setEncoding('binary');\n\n            response.on('data', chunk => {\n                body += chunk;\n            });\n\n            response.on('end', () => {\n                response.rawEncoded = body;\n                this.decode(new Buffer(body, 'binary'), response, (err, body) => {\n                    if (err) {\n                        this.fireError(err, response);\n                        return;\n                    }\n                    response.raw = body;\n                    body = this.iconv(body, response);\n                    this.encode(body, response, (err, body) => {\n                        if (err) {\n                            this.fireError(err, response);\n                        }\n                        else {\n                            this.fireSuccess(body, response);\n                        }\n                    });\n                });\n            });\n        }\n    }\n    decode(body, response, callback) {\n        const decoder = response.headers['content-encoding'];\n\n        if (decoder in decoders) {\n            decoders[decoder].call(response, body, callback);\n        }\n        else {\n            callback(null, body);\n        }\n    }\n    iconv(body, response) {\n        let charset = response.headers['content-type'];\n        if (charset) {\n            charset = /\\bcharset=(.+)(?:;|$)/i.exec(charset);\n            if (charset) {\n                charset = charset[1].trim().toUpperCase();\n                if (charset != 'UTF-8') {\n                    try {\n                        return iconv.decode(body, charset);\n                    }\n                    catch (err) {}\n                }\n            }\n        }\n        return body;\n    }\n    encode(body, response, callback) {\n        if (this.options.decoding == 'buffer') {\n            callback(null, body);\n        }\n        else {\n            body = body.toString(this.options.decoding);\n            if (this.options.parser) {\n                this.options.parser.call(response, body, callback);\n            }\n            else {\n                callback(null, body);\n            }\n        }\n    }\n    fireError(err, response) {\n        this.fireCancelTimeout();\n        this.emit('error', err, response);\n        this.emit('complete', err, response);\n    }\n    fireCancelTimeout() {\n        if (this.options.timeout) {\n            clearTimeout(this.options.timeoutFn);\n        }\n    }\n    fireTimeout(err) {\n        this.emit('timeout', err);\n        this.aborted = true;\n        this.timedout = true;\n        this.request.abort();\n    }\n    fireSuccess(body, response) {\n        if (parseInt(response.statusCode,10) >= 400) {\n            this.emit('fail', body, response);\n        }\n        else {\n            this.emit('success', body, response);\n        }\n        this.emit(response.statusCode.toString().replace(/\\d{2}$/, 'XX'), body, response);\n        this.emit(response.statusCode.toString(), body, response);\n        this.emit('complete', body, response);\n    }\n    makeRequest() {\n        const timeoutMs = this.options.timeout;\n        if (timeoutMs) {\n            this.options.timeoutFn = setTimeout(() => {\n                this.fireTimeout(timeoutMs);\n            }, timeoutMs);\n        }\n        this.request.on('response', response => {\n            this.fireCancelTimeout();\n            this.emit('response', response);\n            this.responseHandler(response);\n        }).on('error', err => {\n            this.fireCancelTimeout();\n            if (!this.aborted) {\n                this.fireError(err, null);\n            }\n        });\n    }\n    reRetry() {\n        this.request.removeAllListeners().on('error', () => {});\n        if (this.request.finished) {\n            this.request.abort();\n        }\n        this.prepare(this.url.href, this.options); // reusing request object to handle recursive calls and remember listeners\n        this.run();\n    }\n    async run() {\n        if (this.options.multipart) {\n            await multipart.write(this.request, this.options.data, () => {\n                this.request.end();\n            });\n        }\n        else {\n            if (this.options.data) {\n                this.request.write(this.options.data, this.options.encoding || 'utf8');\n            }\n            this.request.end();\n        }\n\n        return this;\n    }\n    abort(err) {\n        if (err) {\n            if (typeof err == 'string') {\n                err = new Error(err);\n            }\n            else if (!(err instanceof Error)) {\n                err = new Error('AbortError');\n            }\n            err.type = 'abort';\n        }\n        else {\n            err = null;\n        }\n\n        this.request.on('close', () => {\n            if (err) {\n                this.fireError(err, null);\n            }\n            else {\n                this.emit('complete', null, null);\n            }\n        });\n\n        this.aborted = true;\n        this.request.abort();\n        this.emit('abort', err);\n        return this;\n    }\n    retry(timeout) {\n        timeout = parseInt(timeout,10);\n        const fn = this.reRetry.bind(this);\n        if (!isFinite(timeout) || timeout <= 0) {\n            process.nextTick(fn, timeout);\n        }\n        else {\n            setTimeout(fn, timeout);\n        }\n        return this;\n    }\n}\n\nconst logger=new Logger('xrest');\n\nexport function emit(eventString,options={}) {\n    let [event,path,...middlewares]=eventString.split(' ');\n    let middleFunctions=[];\n    for(let middleware of middlewares){\n        if(middleware.toUpperCase()!==middleware){\n            logger.warn('Upper case is preffered for middleware names! (Got: %s)',event);\n            middleware=middleware.toUpperCase();\n        }\n        if(!~POSSIBLE_MIDDLEWARES.indexOf(middleware))\n            throw new Error('Unknown middleware: '+middleware);\n        middleFunctions.push(middleware);\n    }\n    if(event.toUpperCase()!==event){\n        logger.warn('Upper case is preffered for event names! (Got: %s)',event);\n        event=event.toUpperCase();\n    }\n    if(!~POSSIBLE_EVENTS.indexOf(event)){\n        throw new Error('Unknown event: '+event+', possible events are '+POSSIBLE_EVENTS.join(', ')+'!');\n    }\n    options.method = event;\n    const request=new Request(path,options);\n    if(~middleFunctions.indexOf('STREAM')){\n        logger.debug('Streaming');\n        return new Promise((res,rej)=>{\n            request.run();\n            request.on('timeout',(ms)=>{\n                rej(new Error('Timeout: '+ms));\n            });\n            request.on('response',(response)=>{\n                res(response);\n            });\n        });\n    }else{\n        return new Promise((res,rej)=>{\n            request.run();\n            request.on('timeout',(ms)=>{\n                rej(new Error('Timeout: '+ms));\n            });\n            request.on('complete',(result,response)=>{\n                if(result instanceof Error) {\n                    rej(result);\n                    return;\n                }\n                response.body=result;\n                res(response);\n            });\n        });\n    }\n}\nexport default class XRest{\n    baseUrl;\n    defaultOptions;\n    constructor(url,defaultOptions){\n        logger.debug('new XRest(%s)',url);\n        this.baseUrl=url;\n        this.defaultOptions=defaultOptions;\n    }\n    emit(eventString,options){\n        let [event,path,...middlewares]=eventString.split(' ');\n        path=resolve(this.baseUrl, path);\n        return emit([event,path,...middlewares].join(' '),options);\n    }\n}"],"sourceRoot":"/source/"} |
197
multipart.js
@@ -1,29 +0,28 @@ | ||
import { | ||
basename | ||
} | ||
from 'path'; | ||
import { | ||
open, | ||
read, | ||
close | ||
} | ||
from '@meteor-it/fs'; | ||
export const DEFAULT_BOUNDARY = '84921024METEORITXREST74819204'; | ||
export class Stream { | ||
stream; | ||
string; | ||
"use strict"; | ||
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { | ||
return new (P || (P = Promise))(function (resolve, reject) { | ||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } | ||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } | ||
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); } | ||
step((generator = generator.apply(thisArg, _arguments || [])).next()); | ||
}); | ||
}; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
const path_1 = require("path"); | ||
const fs_1 = require("@meteor-it/fs"); | ||
exports.DEFAULT_BOUNDARY = '84921024METEORITXREST74819204'; | ||
class Stream { | ||
constructor(stream) { | ||
if (this._isString(stream)) { | ||
this.string = ''; | ||
} | ||
this.stream = stream; | ||
if (this._isString(stream)) { | ||
this.string = ''; | ||
} | ||
this.stream = stream; | ||
} | ||
write(data) { | ||
if (this.string != undefined) { | ||
this.string += data; | ||
} else { | ||
this.stream.write(data, 'binary'); | ||
} | ||
if (this.string != undefined) { | ||
this.string += data; | ||
} | ||
else { | ||
this.stream.write(data, 'binary'); | ||
} | ||
} | ||
@@ -34,12 +33,7 @@ _isString(obj) { | ||
} | ||
export class File { | ||
path; | ||
filename; | ||
fileSize; | ||
encoding; | ||
contentType; | ||
exports.Stream = Stream; | ||
class File { | ||
constructor(path, filename, fileSize, encoding, contentType) { | ||
this.path = path; | ||
this.filename = filename || basename(path); | ||
this.filename = filename || path_1.basename(path); | ||
this.fileSize = fileSize; | ||
@@ -50,12 +44,8 @@ this.encoding = encoding || 'binary'; | ||
} | ||
export class FileStream { | ||
filename; | ||
fileSize; | ||
encoding; | ||
contentType; | ||
exports.File = File; | ||
class FileStream { | ||
constructor(stream, filename, dataLength, encoding, contentType) { | ||
if(!dataLength || dataLength!==dataLength) | ||
if (!dataLength || dataLength !== dataLength) | ||
throw new Error('Building FileStream without dataLength!'); | ||
this.stream=stream; | ||
this.stream = stream; | ||
this.filename = filename; | ||
@@ -67,8 +57,4 @@ this.fileSize = dataLength; | ||
} | ||
export class Data { | ||
filename; | ||
contentType; | ||
data; | ||
exports.FileStream = FileStream; | ||
class Data { | ||
constructor(filename, contentType, data) { | ||
@@ -80,7 +66,4 @@ this.filename = filename; | ||
} | ||
export class Part { | ||
name; | ||
value; | ||
boundary; | ||
exports.Data = Data; | ||
class Part { | ||
constructor(name, value, boundary) { | ||
@@ -91,7 +74,5 @@ this.name = name; | ||
} | ||
//returns the Content-Disposition header | ||
header() { | ||
let header; | ||
if (this.value.data) { | ||
@@ -103,3 +84,3 @@ header = `Content-Disposition: form-data; name='${this.name}'; filename='${this.value.filename}'\r\nContent-Length: ${this.value.data.length}\r\nContent-Type: ${this.value.contentType}`; | ||
} | ||
else if(this.value instanceof FileStream) { | ||
else if (this.value instanceof FileStream) { | ||
header = `Content-Disposition: form-data; name='${this.name}'; filename='${this.value.filename}'\r\nContent-Length: ${this.value.fileSize}\r\nContent-Type: ${this.value.contentType}`; | ||
@@ -110,6 +91,4 @@ } | ||
} | ||
return `--${this.boundary}\r\n${header}\r\n\r\n`; | ||
} | ||
//calculates the size of the Part | ||
@@ -135,19 +114,17 @@ sizeOf() { | ||
} | ||
// Writes the Part out to a writable stream that supports the write(data) method | ||
write(stream) { | ||
return new Promise(async (resolve,reject)=>{ | ||
if(!stream.on) | ||
if(stream.stream) | ||
stream=stream.stream; | ||
return new Promise((resolve, reject) => __awaiter(this, void 0, void 0, function* () { | ||
if (!stream.on) | ||
if (stream.stream) | ||
stream = stream.stream; | ||
//first write the Content-Disposition | ||
stream.write(this.header()); | ||
//Now write out the body of the Part | ||
if (this.value instanceof File) { | ||
let fd = await open(this.value.path, 'r', '0666'); | ||
let fd = yield fs_1.open(this.value.path, 'r', '0666'); | ||
let position = 0; | ||
let moreData = true; | ||
while (moreData) { | ||
let chunk = await read(fd, 4096, position, 'binary'); | ||
let chunk = yield fs_1.read(fd, 4096, position, 'binary'); | ||
stream.write(chunk); | ||
@@ -160,3 +137,3 @@ position += 4096; | ||
stream.write('\r\n'); | ||
close(fd); | ||
fs_1.close(fd); | ||
moreData = false; | ||
@@ -166,13 +143,13 @@ resolve(); | ||
} | ||
} else if(this.value instanceof FileStream){ | ||
this.value.stream.on('end', () =>{ | ||
} | ||
else if (this.value instanceof FileStream) { | ||
this.value.stream.on('end', () => { | ||
stream.write('\r\n'); | ||
resolve(); | ||
}); | ||
let s=this.value.stream.pipe(stream,{ | ||
end:false // Do not end writing streams, may be there is more data incoming | ||
let s = this.value.stream.pipe(stream, { | ||
end: false // Do not end writing streams, may be there is more data incoming | ||
}); | ||
} else if (this.value instanceof Data) { | ||
} | ||
else if (this.value instanceof Data) { | ||
stream.write(this.value.data); | ||
@@ -185,19 +162,14 @@ stream.write('\r\n'); | ||
resolve(); | ||
} | ||
}) | ||
} | ||
})); | ||
} | ||
} | ||
export class MultiPartRequest { | ||
encoding; | ||
boundary; | ||
data; | ||
partNames; | ||
exports.Part = Part; | ||
class MultiPartRequest { | ||
constructor(data, boundary) { | ||
this.encoding = 'binary'; | ||
this.boundary = boundary || DEFAULT_BOUNDARY; | ||
this.boundary = boundary || exports.DEFAULT_BOUNDARY; | ||
this.data = data; | ||
this.partNames = this._partNames(); | ||
} | ||
_partNames() { | ||
@@ -210,34 +182,37 @@ const partNames = []; | ||
} | ||
async write(stream) { | ||
let partCount = 0; | ||
// wrap the stream in our own Stream object | ||
// See the Stream function above for the benefits of this | ||
stream = new Stream(stream); | ||
while (true) { | ||
const partName = this.partNames[partCount]; | ||
const part = new Part(partName, this.data[partName], this.boundary); | ||
await part.write(stream); | ||
partCount++; | ||
if (partCount >= this.partNames.length) { | ||
stream.write(`--${this.boundary}--\r\n`); | ||
return stream.string || ''; | ||
write(stream) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
let partCount = 0; | ||
// wrap the stream in our own Stream object | ||
// See the Stream function above for the benefits of this | ||
stream = new Stream(stream); | ||
while (true) { | ||
const partName = this.partNames[partCount]; | ||
const part = new Part(partName, this.data[partName], this.boundary); | ||
yield part.write(stream); | ||
partCount++; | ||
if (partCount >= this.partNames.length) { | ||
stream.write(`--${this.boundary}--\r\n`); | ||
return stream.string || ''; | ||
} | ||
} | ||
} | ||
}); | ||
} | ||
} | ||
export function sizeOf(parts, boundary=DEFAULT_BOUNDARY) { | ||
exports.MultiPartRequest = MultiPartRequest; | ||
function sizeOf(parts, boundary = exports.DEFAULT_BOUNDARY) { | ||
let totalSize = 0; | ||
for (let name in parts) | ||
totalSize += new Part(name, parts[name], boundary).sizeOf(); | ||
return totalSize + boundary.length + 6; | ||
for (let name in parts) | ||
totalSize += new Part(name, parts[name], boundary).sizeOf(); | ||
return totalSize + boundary.length + 6; | ||
} | ||
export async function write(stream, data, callback, boundary) { | ||
let r = new MultiPartRequest(data, boundary); | ||
await r.write(stream); | ||
return r; | ||
} | ||
exports.sizeOf = sizeOf; | ||
function write(stream, data, callback, boundary) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
let r = new MultiPartRequest(data, boundary); | ||
yield r.write(stream); | ||
return r; | ||
}); | ||
} | ||
exports.write = write; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["xrest/multipart.js"],"names":[],"mappings":"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","file":"xrest/multipart.js","sourcesContent":["import {\n    basename\n}\nfrom 'path';\nimport {\n    open,\n    read,\n    close\n}\nfrom '@meteor-it/fs';\n\nexport const DEFAULT_BOUNDARY = '84921024METEORITXREST74819204';\n\nexport class Stream {\n    stream;\n    string;\n    constructor(stream) {\n    \tif (this._isString(stream)) {\n    \t\tthis.string = '';\n    \t}\n    \tthis.stream = stream;\n    }\n    write(data) {\n    \tif (this.string != undefined) {\n    \t\tthis.string += data;\n    \t} else {\n    \t\tthis.stream.write(data, 'binary');\n    \t}\n    }\n    _isString(obj) {\n        return !!(obj === '' || (obj && obj.charCodeAt && obj.substr));\n    }\n}\n\nexport class File {\n    path;\n    filename;\n    fileSize;\n    encoding;\n    contentType;\n    constructor(path, filename, fileSize, encoding, contentType) {\n        this.path = path;\n        this.filename = filename || basename(path);\n        this.fileSize = fileSize;\n        this.encoding = encoding || 'binary';\n        this.contentType = contentType || 'application/octet-stream';\n    }\n}\n\nexport class FileStream {\n    filename;\n    fileSize;\n    encoding;\n    contentType;\n    constructor(stream, filename, dataLength, encoding, contentType) {\n        if(!dataLength || dataLength!==dataLength)\n            throw new Error('Building FileStream without dataLength!');\n        this.stream=stream;\n        this.filename = filename;\n        this.fileSize = dataLength;\n        this.encoding = encoding || 'binary';\n        this.contentType = contentType || 'application/octet-stream';\n    }\n}\n\nexport class Data {\n    filename;\n    contentType;\n    data;\n\n    constructor(filename, contentType, data) {\n        this.filename = filename;\n        this.contentType = contentType || 'application/octet-stream';\n        this.data = data;\n    }\n}\n\nexport class Part {\n    name;\n    value;\n    boundary;\n    constructor(name, value, boundary) {\n        this.name = name;\n        this.value = value;\n        this.boundary = boundary;\n    }\n\n    //returns the Content-Disposition header\t\t\n    header() {\n        let header;\n\n        if (this.value.data) {\n            header = `Content-Disposition: form-data; name='${this.name}'; filename='${this.value.filename}'\\r\\nContent-Length: ${this.value.data.length}\\r\\nContent-Type: ${this.value.contentType}`;\n        }\n        else if (this.value instanceof File) {\n            header = `Content-Disposition: form-data; name='${this.name}'; filename='${this.value.filename}'\\r\\nContent-Length: ${this.value.fileSize}\\r\\nContent-Type: ${this.value.contentType}`;\n        }\n        else if(this.value instanceof FileStream) {\n            header = `Content-Disposition: form-data; name='${this.name}'; filename='${this.value.filename}'\\r\\nContent-Length: ${this.value.fileSize}\\r\\nContent-Type: ${this.value.contentType}`;\n        }\n        else {\n            header = `Content-Disposition: form-data; name='${this.name}'`;\n        }\n\n        return `--${this.boundary}\\r\\n${header}\\r\\n\\r\\n`;\n    }\n\n    //calculates the size of the Part\n    sizeOf() {\n        let valueSize;\n        if (this.value instanceof File) {\n            valueSize = this.value.fileSize;\n        }\n        else if (this.value instanceof FileStream) {\n            valueSize = this.value.fileSize;\n        }\n        else if (this.value.data) {\n            valueSize = this.value.data.length;\n        }\n        else if (typeof this.value === 'number') {\n            valueSize = this.value.toString().length;\n        }\n        else {\n            valueSize = this.value.length;\n        }\n        return valueSize + this.header().length + 2;\n    }\n\n    // Writes the Part out to a writable stream that supports the write(data) method\n    write(stream) {\n        return new Promise(async (resolve,reject)=>{\n            if(!stream.on)\n            if(stream.stream)\n                stream=stream.stream;\n            //first write the Content-Disposition\n            stream.write(this.header());\n    \n            //Now write out the body of the Part\n            if (this.value instanceof File) {\n                let fd = await open(this.value.path, 'r', '0666');\n                let position = 0;\n                let moreData = true;\n                while (moreData) {\n                    let chunk = await read(fd, 4096, position, 'binary');\n                    stream.write(chunk);\n                    position += 4096;\n                    if (chunk) {\n                        moreData = true;\n                    }\n                    else {\n                        stream.write('\\r\\n');\n                        close(fd);\n                        moreData = false;\n                        resolve();\n                    }\n                }\n            } else if(this.value instanceof FileStream){\n\n                this.value.stream.on('end', () =>{\n                    stream.write('\\r\\n');\n                    resolve();\n                });\n                \n                let s=this.value.stream.pipe(stream,{\n                    end:false // Do not end writing streams, may be there is more data incoming\n                });\n            } else if (this.value instanceof Data) {\n                stream.write(this.value.data);\n                stream.write('\\r\\n');\n                resolve();\n            }\n            else {\n                stream.write(`${this.value}\\r\\n`);\n                resolve();\n            }  \n        })\n    }\n}\n\nexport class MultiPartRequest {\n    encoding;\n    boundary;\n    data;\n    partNames;\n    constructor(data, boundary) {\n        this.encoding = 'binary';\n        this.boundary = boundary || DEFAULT_BOUNDARY;\n        this.data = data;\n        this.partNames = this._partNames();\n    }\n\n    _partNames() {\n        const partNames = [];\n        for (const name in this.data) {\n            partNames.push(name);\n        }\n        return partNames;\n    }\n\n    async write(stream) {\n        let partCount = 0;\n\n        // wrap the stream in our own Stream object\n        // See the Stream function above for the benefits of this\n        stream = new Stream(stream);\n        while (true) {\n            const partName = this.partNames[partCount];\n            const part = new Part(partName, this.data[partName], this.boundary);\n            await part.write(stream);\n            partCount++;\n            if (partCount >= this.partNames.length) {\n                stream.write(`--${this.boundary}--\\r\\n`);\n\n                return stream.string || '';\n            }\n        }\n    }\n}\n\n\nexport function sizeOf(parts, boundary=DEFAULT_BOUNDARY) {\n    let totalSize = 0;\n  \tfor (let name in parts) \n  \t    totalSize += new Part(name, parts[name], boundary).sizeOf();\n  \treturn totalSize + boundary.length + 6;\n}\nexport async function write(stream, data, callback, boundary) {\n    let r = new MultiPartRequest(data, boundary);\n    await r.write(stream);\n    return r;\n}"],"sourceRoot":"/source/"} |
{ | ||
"name": "@meteor-it/xrest", | ||
"version": "0.2.1", | ||
"version": "0.3.1", | ||
"description": "I", | ||
@@ -5,0 +5,0 @@ "main": "index.js", |
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Network access
Supply chain riskThis module accesses the network.
Found 2 instances in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Network access
Supply chain riskThis module accesses the network.
Found 2 instances in 1 package
81349
833