@nlabs/reaktor 0.5.3 → 0.5.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -60,13 +60,11 @@ var import_utils = __toModule(require("@nlabs/utils"));
60
60
  var import_arangojs = __toModule(require("arangojs"));
61
61
  var import_googleapis = __toModule(require("googleapis"));
62
62
  var import_isEmpty = __toModule(require("lodash/isEmpty"));
63
- var request = __toModule(require("request-promise"));
64
63
  var import_config = __toModule(require("../config"));
65
64
  var import_analytics = __toModule(require("../utils/analytics"));
66
65
  var import_images = __toModule(require("./images"));
67
66
  var import_posts = __toModule(require("./posts"));
68
67
  const youtube = import_googleapis.google.youtube({ auth: import_config.Config.get("google.key"), version: "v3" });
69
- request.defaults({ encoding: null });
70
68
  const addFile = (context, item = {}) => {
71
69
  const { database, session: { userId: sessionId, userAccess } } = context;
72
70
  const {
@@ -151,7 +149,7 @@ const addFile = (context, item = {}) => {
151
149
  return saveToDb(insert);
152
150
  };
153
151
  if (isUrl) {
154
- return request.get({ encoding: null, uri: url }).then((body) => uploadFile(Buffer.from(body, "binary"), formatType)).catch(() => {
152
+ return (0, import_rip_hunter.get)(url).then((body) => uploadFile(Buffer.from(body, "binary"), formatType)).catch(() => {
155
153
  throw new import_analytics.UserError("file_request");
156
154
  });
157
155
  } else if (item.base64 !== "") {
@@ -319,4 +317,4 @@ const decodeBase64 = (dataString) => {
319
317
  linkFiles,
320
318
  updateFiles
321
319
  });
322
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/files.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {get as httpGet} from '@nlabs/rip-hunter';\nimport {createHash, parseId, parseString} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {google} from 'googleapis';\nimport isEmpty from 'lodash/isEmpty';\nimport * as request from 'request-promise';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {FileType} from '../types/files';\nimport {UserError} from '../utils/analytics';\nimport {resizeSaveImage} from './images';\nimport {createPostEdge} from './posts';\n\nconst youtube = google.youtube({auth: Config.get('google.key'), version: 'v3'});\nrequest.defaults({encoding: null});\n\n// const eventCategory: string = 'files';\n\n// Upload file\nexport const addFile = (context: ApiContext, item: FileType = {}): Promise<FileType> => {\n  const {database, session: {userId: sessionId, userAccess}} = context;\n  const {\n    description,\n    fileId,\n    fileType,\n    name,\n    url\n  } = item;\n\n  // Id\n  const formatFileId: string = fileId ? parseId(fileId) : createHash(`file-${sessionId}`);\n\n  // Name\n  const isUrl: boolean = isEmpty(url);\n\n  // If no name, get it from url path\n  let formatName: string = parseString(name, 160);\n  let formatType: string = parseString(fileType, 16);\n\n  if(formatName === '' && isUrl) {\n    formatName = url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  if(formatType === '') {\n    const nameArr: string[] = formatName.split('.');\n    const ext: string = nameArr[nameArr.length - 1];\n\n    switch(ext) {\n      case 'jpeg':\n      case 'jpg':\n        formatType = 'image/jpeg';\n        break;\n      case 'png':\n        formatType = 'image/png';\n        break;\n      case 'zip':\n        formatType = 'application/zip';\n        break;\n      default:\n        break;\n    }\n  }\n\n  let isImage: boolean;\n\n  switch(formatType) {\n    case 'image/jpeg':\n    case 'image/png':\n      isImage = true;\n      break;\n    default:\n      isImage = false;\n      break;\n  }\n\n  // Description\n  const formatDesc: string = parseString(description, 500);\n\n  // Only allow file uploads to premium users\n  if(!isImage && userAccess !== 2) {\n    throw new UserError('account_restriction');\n  }\n\n  const saveToDb = (insert: FileType) => {\n    const aqlQry: AqlQuery = aql`INSERT ${insert} IN files RETURN NEW`;\n\n    return database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((file = {}) => file)\n      .catch((error: Error) => {\n        throw error;\n      });\n  };\n\n  const uploadFile = (buf: Buffer, uploadType: string) => {\n    const now: number = Date.now();\n\n    // If image, resize and create a thumbnail\n    if(isImage) {\n      return resizeSaveImage(context, formatFileId, buf, uploadType)\n        .then((resizedImage: FileType) => {\n          const insert: FileType = {\n            ...resizedImage,\n            _key: formatFileId,\n            added: now,\n            description: formatDesc,\n            fileType: formatType,\n            modified: now,\n            name: formatName,\n            userId: sessionId\n          };\n\n          return saveToDb(insert);\n        })\n        .catch((error: Error) => {\n          throw error;\n        });\n    }\n    const insert: FileType = {\n      _key: formatFileId,\n      added: now,\n      description: formatDesc,\n      fileType: formatType,\n      modified: now,\n      name: formatName,\n      userId: sessionId\n    };\n\n    return saveToDb(insert);\n  };\n\n  // If file is a url path, download the file and save\n  if(isUrl) {\n    return request.get({encoding: null, uri: url})\n      .then((body) => uploadFile(Buffer.from(body, 'binary'), formatType))\n      .catch(() => {\n        throw new UserError('file_request');\n      });\n  } else if(item.base64 !== '') {\n    const buffer: Buffer = Buffer.from(item.base64);\n    return uploadFile(buffer, formatType);\n  }\n  throw new Error('file_required');\n};\n\n// Giphy\nexport const getGiphyTrends = (context: ApiContext, limit: number = 30): Promise<any[]> => {\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/trending?api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return httpGet(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getGiphySearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> => {\n  const formatQuery: string = encodeURI(query);\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/search?q=${formatQuery}&api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return fetch(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getYouTubeTrends = (context: ApiContext, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.videos.list({\n      chart: 'mostPopular',\n      maxResults: limit,\n      part: ['snippet'],\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const list = data.items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n\nexport const getYouTubeSearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.search.list({\n      maxResults: limit,\n      part: ['snippet'],\n      // eslint-disable-next-line\n      q: query,\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const {items} = data;\n        const list = items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n// Files\nexport const getPathUserFiles = (userId: string, filename: string): string => `users/${userId}/files/${filename}`;\n\nexport const getUrlUserFiles = (userId: string, filename: string, dir: string = 'files', type: string = 'profile'): string => {\n  if(filename) {\n    return `https://box.${Config.get('app.url')}/users/${userId}/${dir}/${filename}`;\n  }\n\n  if(type === 'profile') {\n    return `https://box.${Config.get('app.url')}/defaults/user_bk.jpg`;\n  }\n\n  return `https://box.${Config.get('app.url')}/defaults/user_wh.jpg`;\n};\n\nexport const createFile = (db: Database, file: FileType): Promise<FileType> => {\n  const {fileId} = file;\n  const insert: any = {\n    _key: fileId,\n    added: Date.now()\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${fileId}}\n    INSERT ${insert}\n    UPDATE {}\n    IN files RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((updatedFile: FileType = {}) => updatedFile)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const linkFiles = (db: Database, files: FileType[], postId: string): Promise<any> =>\n  Promise.all(\n    files.map((file: FileType) => createFile(db, file)\n      .then((file: FileType) => createPostEdge(db, file, postId)))\n  );\n\nexport const updateFiles = (db: Database, postId: string, files: FileType[]): Promise<any> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n\n  return edgeCollection.inEdges(postId)\n    .then((edges: any) => {\n      if(edges.length) {\n        // Remove linked edges\n        return Promise.all(\n          edges.map((edge) => {\n            const {_key: edgeKey} = edge;\n            const aqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN isPosted`;\n\n            return db.query(aqlQry).catch((error: Error) => {\n              throw error;\n            });\n          }))\n          .then(() => {\n            if(files.length) {\n              // Link files\n              return linkFiles(db, files, postId).then(() => files);\n            }\n            return files;\n          });\n      } else if(files.length) {\n        // Link files\n        return linkFiles(db, files, postId).then(() => files);\n      }\n      return files;\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const encodeBase64 = (buffer: Buffer): string => Buffer.from(buffer).toString('base64');\n\nexport const decodeBase64 = (dataString: string): object => {\n  const getData = (str: string) => str.match(/^data:([A-Za-z-+\\/]+);base64,(.+)$/) || [];\n  // const getData = (base64: string) => base64.substr(base64.indexOf(',') + 1);\n  let matches = getData(dataString);\n\n  if(matches.length !== 3) {\n    // If invalid make sure we don't need to decode\n    matches = getData(decodeURIComponent(dataString));\n\n    // Check it again.\n    if(matches.length !== 3) {\n      throw Error('Invalid input string');\n    }\n  }\n\n  return {\n    data: Buffer.from(matches[2], 'base64'),\n    type: matches[1]\n  };\n};\n"],
  "mappings": "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",
  "names": []
}

320
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/files.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {get as httpGet} from '@nlabs/rip-hunter';\nimport {createHash, parseId, parseString} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {google} from 'googleapis';\nimport isEmpty from 'lodash/isEmpty';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {FileType} from '../types/files';\nimport {UserError} from '../utils/analytics';\nimport {resizeSaveImage} from './images';\nimport {createPostEdge} from './posts';\n\nconst youtube = google.youtube({auth: Config.get('google.key'), version: 'v3'});\n\n// const eventCategory: string = 'files';\n\n// Upload file\nexport const addFile = (context: ApiContext, item: FileType = {}): Promise<FileType> => {\n  const {database, session: {userId: sessionId, userAccess}} = context;\n  const {\n    description,\n    fileId,\n    fileType,\n    name,\n    url\n  } = item;\n\n  // Id\n  const formatFileId: string = fileId ? parseId(fileId) : createHash(`file-${sessionId}`);\n\n  // Name\n  const isUrl: boolean = isEmpty(url);\n\n  // If no name, get it from url path\n  let formatName: string = parseString(name, 160);\n  let formatType: string = parseString(fileType, 16);\n\n  if(formatName === '' && isUrl) {\n    formatName = url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  if(formatType === '') {\n    const nameArr: string[] = formatName.split('.');\n    const ext: string = nameArr[nameArr.length - 1];\n\n    switch(ext) {\n      case 'jpeg':\n      case 'jpg':\n        formatType = 'image/jpeg';\n        break;\n      case 'png':\n        formatType = 'image/png';\n        break;\n      case 'zip':\n        formatType = 'application/zip';\n        break;\n      default:\n        break;\n    }\n  }\n\n  let isImage: boolean;\n\n  switch(formatType) {\n    case 'image/jpeg':\n    case 'image/png':\n      isImage = true;\n      break;\n    default:\n      isImage = false;\n      break;\n  }\n\n  // Description\n  const formatDesc: string = parseString(description, 500);\n\n  // Only allow file uploads to premium users\n  if(!isImage && userAccess !== 2) {\n    throw new UserError('account_restriction');\n  }\n\n  const saveToDb = (insert: FileType) => {\n    const aqlQry: AqlQuery = aql`INSERT ${insert} IN files RETURN NEW`;\n\n    return database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((file = {}) => file)\n      .catch((error: Error) => {\n        throw error;\n      });\n  };\n\n  const uploadFile = (buf: Buffer, uploadType: string) => {\n    const now: number = Date.now();\n\n    // If image, resize and create a thumbnail\n    if(isImage) {\n      return resizeSaveImage(context, formatFileId, buf, uploadType)\n        .then((resizedImage: FileType) => {\n          const insert: FileType = {\n            ...resizedImage,\n            _key: formatFileId,\n            added: now,\n            description: formatDesc,\n            fileType: formatType,\n            modified: now,\n            name: formatName,\n            userId: sessionId\n          };\n\n          return saveToDb(insert);\n        })\n        .catch((error: Error) => {\n          throw error;\n        });\n    }\n    const insert: FileType = {\n      _key: formatFileId,\n      added: now,\n      description: formatDesc,\n      fileType: formatType,\n      modified: now,\n      name: formatName,\n      userId: sessionId\n    };\n\n    return saveToDb(insert);\n  };\n\n  // If file is a url path, download the file and save\n  if(isUrl) {\n    return httpGet(url)\n      .then((body) => uploadFile(Buffer.from(body, 'binary'), formatType))\n      .catch(() => {\n        throw new UserError('file_request');\n      });\n  } else if(item.base64 !== '') {\n    const buffer: Buffer = Buffer.from(item.base64);\n    return uploadFile(buffer, formatType);\n  }\n  throw new Error('file_required');\n};\n\n// Giphy\nexport const getGiphyTrends = (context: ApiContext, limit: number = 30): Promise<any[]> => {\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/trending?api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return httpGet(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getGiphySearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> => {\n  const formatQuery: string = encodeURI(query);\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/search?q=${formatQuery}&api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return fetch(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getYouTubeTrends = (context: ApiContext, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.videos.list({\n      chart: 'mostPopular',\n      maxResults: limit,\n      part: ['snippet'],\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const list = data.items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n\nexport const getYouTubeSearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.search.list({\n      maxResults: limit,\n      part: ['snippet'],\n      // eslint-disable-next-line\n      q: query,\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const {items} = data;\n        const list = items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n// Files\nexport const getPathUserFiles = (userId: string, filename: string): string => `users/${userId}/files/${filename}`;\n\nexport const getUrlUserFiles = (userId: string, filename: string, dir: string = 'files', type: string = 'profile'): string => {\n  if(filename) {\n    return `https://box.${Config.get('app.url')}/users/${userId}/${dir}/${filename}`;\n  }\n\n  if(type === 'profile') {\n    return `https://box.${Config.get('app.url')}/defaults/user_bk.jpg`;\n  }\n\n  return `https://box.${Config.get('app.url')}/defaults/user_wh.jpg`;\n};\n\nexport const createFile = (db: Database, file: FileType): Promise<FileType> => {\n  const {fileId} = file;\n  const insert: any = {\n    _key: fileId,\n    added: Date.now()\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${fileId}}\n    INSERT ${insert}\n    UPDATE {}\n    IN files RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((updatedFile: FileType = {}) => updatedFile)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const linkFiles = (db: Database, files: FileType[], postId: string): Promise<any> =>\n  Promise.all(\n    files.map((file: FileType) => createFile(db, file)\n      .then((file: FileType) => createPostEdge(db, file, postId)))\n  );\n\nexport const updateFiles = (db: Database, postId: string, files: FileType[]): Promise<any> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n\n  return edgeCollection.inEdges(postId)\n    .then((edges: any) => {\n      if(edges.length) {\n        // Remove linked edges\n        return Promise.all(\n          edges.map((edge) => {\n            const {_key: edgeKey} = edge;\n            const aqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN isPosted`;\n\n            return db.query(aqlQry).catch((error: Error) => {\n              throw error;\n            });\n          }))\n          .then(() => {\n            if(files.length) {\n              // Link files\n              return linkFiles(db, files, postId).then(() => files);\n            }\n            return files;\n          });\n      } else if(files.length) {\n        // Link files\n        return linkFiles(db, files, postId).then(() => files);\n      }\n      return files;\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const encodeBase64 = (buffer: Buffer): string => Buffer.from(buffer).toString('base64');\n\nexport const decodeBase64 = (dataString: string): object => {\n  const getData = (str: string) => str.match(/^data:([A-Za-z-+\\/]+);base64,(.+)$/) || [];\n  // const getData = (base64: string) => base64.substr(base64.indexOf(',') + 1);\n  let matches = getData(dataString);\n\n  if(matches.length !== 3) {\n    // If invalid make sure we don't need to decode\n    matches = getData(decodeURIComponent(dataString));\n\n    // Check it again.\n    if(matches.length !== 3) {\n      throw Error('Invalid input string');\n    }\n  }\n\n  return {\n    data: Buffer.from(matches[2], 'base64'),\n    type: matches[1]\n  };\n};\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAIA,wBAA6B;AAC7B,mBAA+C;AAC/C,sBAA4B;AAI5B,wBAAqB;AACrB,qBAAoB;AAEpB,oBAAqB;AAGrB,uBAAwB;AACxB,oBAA8B;AAC9B,mBAA6B;AAE7B,MAAM,UAAU,yBAAO,QAAQ,EAAC,MAAM,qBAAO,IAAI,eAAe,SAAS;AAKlE,MAAM,UAAU,CAAC,SAAqB,OAAiB,OAA0B;AACtF,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,WAAW,iBAAe;AAC7D,QAAM;AAAA,IACJ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,MACE;AAGJ,QAAM,eAAuB,SAAS,0BAAQ,UAAU,6BAAW,QAAQ;AAG3E,QAAM,QAAiB,4BAAQ;AAG/B,MAAI,aAAqB,8BAAY,MAAM;AAC3C,MAAI,aAAqB,8BAAY,UAAU;AAE/C,MAAG,eAAe,MAAM,OAAO;AAC7B,iBAAa,IAAI,UAAU,IAAI,YAAY,OAAO;AAAA;AAGpD,MAAG,eAAe,IAAI;AACpB,UAAM,UAAoB,WAAW,MAAM;AAC3C,UAAM,MAAc,QAAQ,QAAQ,SAAS;AAE7C,YAAO;AAAA,WACA;AAAA,WACA;AACH,qBAAa;AACb;AAAA,WACG;AACH,qBAAa;AACb;AAAA,WACG;AACH,qBAAa;AACb;AAAA;AAEA;AAAA;AAAA;AAIN,MAAI;AAEJ,UAAO;AAAA,SACA;AAAA,SACA;AACH,gBAAU;AACV;AAAA;AAEA,gBAAU;AACV;AAAA;AAIJ,QAAM,aAAqB,8BAAY,aAAa;AAGpD,MAAG,CAAC,WAAW,eAAe,GAAG;AAC/B,UAAM,IAAI,2BAAU;AAAA;AAGtB,QAAM,WAAW,CAAC,WAAqB;AACrC,UAAM,SAAmB,6BAAa;AAEtC,WAAO,SAAS,MAAM,QACnB,KAAK,CAAC,WAAwB,OAAO,QACrC,KAAK,CAAC,OAAO,OAAO,MACpB,MAAM,CAAC,UAAiB;AACvB,YAAM;AAAA;AAAA;AAIZ,QAAM,aAAa,CAAC,KAAa,eAAuB;AACtD,UAAM,MAAc,KAAK;AAGzB,QAAG,SAAS;AACV,aAAO,mCAAgB,SAAS,cAAc,KAAK,YAChD,KAAK,CAAC,iBAA2B;AAChC,cAAM,UAAmB,iCACpB,eADoB;AAAA,UAEvB,MAAM;AAAA,UACN,OAAO;AAAA,UACP,aAAa;AAAA,UACb,UAAU;AAAA,UACV,UAAU;AAAA,UACV,MAAM;AAAA,UACN,QAAQ;AAAA;AAGV,eAAO,SAAS;AAAA,SAEjB,MAAM,CAAC,UAAiB;AACvB,cAAM;AAAA;AAAA;AAGZ,UAAM,SAAmB;AAAA,MACvB,MAAM;AAAA,MACN,OAAO;AAAA,MACP,aAAa;AAAA,MACb,UAAU;AAAA,MACV,UAAU;AAAA,MACV,MAAM;AAAA,MACN,QAAQ;AAAA;AAGV,WAAO,SAAS;AAAA;AAIlB,MAAG,OAAO;AACR,WAAO,2BAAQ,KACZ,KAAK,CAAC,SAAS,WAAW,OAAO,KAAK,MAAM,WAAW,aACvD,MAAM,MAAM;AACX,YAAM,IAAI,2BAAU;AAAA;AAAA,aAEhB,KAAK,WAAW,IAAI;AAC5B,UAAM,SAAiB,OAAO,KAAK,KAAK;AACxC,WAAO,WAAW,QAAQ;AAAA;AAE5B,QAAM,IAAI,MAAM;AAAA;AAIX,MAAM,iBAAiB,CAAC,SAAqB,QAAgB,OAAuB;AACzF,QAAM,SAAiB,iDAAiD,qBAAO,IAAI,sBAAsB;AAEzG,SAAO,2BAAQ,QACZ,KAAK,CAAC,QAAkB,IAAI,QAC5B,KAAK,CAAC,SAAS,KAAK,KAAK,IAAI,CAAC,WAAW,EAAC,IAAI,MAAM,QAAQ,WAAU;AACrE,UAAM;AAAA,MACJ;AAAA,MACA,QAAQ;AAAA,QACN,UAAU,EAAC,MAAM,OAAM;AAAA,QACvB,oBAAoB,EAAC,KAAK,QAAQ,OAAM;AAAA,UACtC;AAAA,QACF;AAEJ,WAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA,MAAM;AAAA,MACN;AAAA;AAAA;AAAA;AAKD,MAAM,iBAAiB,CAAC,SAAqB,OAAe,QAAgB,OAAuB;AACxG,QAAM,cAAsB,UAAU;AACtC,QAAM,SAAiB,yCAAyC,uBAAuB,qBAAO,IAAI,sBAAsB;AAExH,SAAO,MAAM,QACV,KAAK,CAAC,QAAkB,IAAI,QAC5B,KAAK,CAAC,SAAS,KAAK,KAAK,IAAI,CAAC,WAAW,EAAC,IAAI,MAAM,QAAQ,WAAU;AACrE,UAAM;AAAA,MACJ;AAAA,MACA,QAAQ;AAAA,QACN,UAAU,EAAC,MAAM,OAAM;AAAA,QACvB,oBAAoB,EAAC,KAAK,QAAQ,OAAM;AAAA,UACtC;AAAA,QACF;AAEJ,WAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA,MAAM;AAAA,MACN;AAAA;AAAA;AAAA;AAKD,MAAM,mBAAmB,CAAC,SAAqB,QAAgB,OACpE,IAAI,QAAQ,CAAC,SAAS,WAAW;AAC/B,UAAQ,OAAO,KAAK;AAAA,IAClB,OAAO;AAAA,IACP,YAAY;AAAA,IACZ,MAAM,CAAC;AAAA,IACP,YAAY;AAAA,KACX,CAAC,OAAc,SAAc;AAC9B,QAAG,OAAO;AACR,cAAQ,MAAM;AACd,aAAO,IAAI,MAAM,MAAM,GAAG;AAAA,eAClB,MAAM;AACd,YAAM,OAAO,KAAK,MAAM,IAAI,CAAC,SAAU;AAAA,QACrC,IAAI,KAAK;AAAA,QACT,OAAO,KAAK,QAAQ,WAAW,KAAK;AAAA,QACpC,MAAM;AAAA,QACN,KAAK,gCAAgC,KAAK;AAAA;AAG5C,cAAQ;AAAA;AAAA;AAAA;AAMT,MAAM,mBAAmB,CAAC,SAAqB,OAAe,QAAgB,OACnF,IAAI,QAAQ,CAAC,SAAS,WAAW;AAC/B,UAAQ,OAAO,KAAK;AAAA,IAClB,YAAY;AAAA,IACZ,MAAM,CAAC;AAAA,IAEP,GAAG;AAAA,IACH,YAAY;AAAA,KACX,CAAC,OAAc,SAAc;AAC9B,QAAG,OAAO;AACR,cAAQ,MAAM;AACd,aAAO,IAAI,MAAM,MAAM,GAAG;AAAA,eAClB,MAAM;AACd,YAAM,EAAC,UAAS;AAChB,YAAM,OAAO,MAAM,IAAI,CAAC,SAAU;AAAA,QAChC,IAAI,KAAK;AAAA,QACT,OAAO,KAAK,QAAQ,WAAW,KAAK;AAAA,QACpC,MAAM;AAAA,QACN,KAAK,gCAAgC,KAAK;AAAA;AAG5C,cAAQ;AAAA;AAAA;AAAA;AAMT,MAAM,mBAAmB,CAAC,QAAgB,aAA6B,SAAS,gBAAgB;AAEhG,MAAM,kBAAkB,CAAC,QAAgB,UAAkB,MAAc,SAAS,OAAe,cAAsB;AAC5H,MAAG,UAAU;AACX,WAAO,eAAe,qBAAO,IAAI,oBAAoB,UAAU,OAAO;AAAA;AAGxE,MAAG,SAAS,WAAW;AACrB,WAAO,eAAe,qBAAO,IAAI;AAAA;AAGnC,SAAO,eAAe,qBAAO,IAAI;AAAA;AAG5B,MAAM,aAAa,CAAC,IAAc,SAAsC;AAC7E,QAAM,EAAC,WAAU;AACjB,QAAM,SAAc;AAAA,IAClB,MAAM;AAAA,IACN,OAAO,KAAK;AAAA;AAGd,QAAM,SAAmB,oCAAoB;AAAA,aAClC;AAAA;AAAA;AAIX,SAAO,GAAG,MAAM,QACb,KAAK,CAAC,WAAwB,OAAO,QACrC,KAAK,CAAC,cAAwB,OAAO,aACrC,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA;AAAA;AAIL,MAAM,YAAY,CAAC,IAAc,OAAmB,WACzD,QAAQ,IACN,MAAM,IAAI,CAAC,SAAmB,WAAW,IAAI,MAC1C,KAAK,CAAC,UAAmB,iCAAe,IAAI,OAAM;AAGlD,MAAM,cAAc,CAAC,IAAc,QAAgB,UAAoC;AAC5F,QAAM,iBAAiC,GAAG,WAAW;AAErD,SAAO,eAAe,QAAQ,QAC3B,KAAK,CAAC,UAAe;AACpB,QAAG,MAAM,QAAQ;AAEf,aAAO,QAAQ,IACb,MAAM,IAAI,CAAC,SAAS;AAClB,cAAM,EAAC,MAAM,YAAW;AACxB,cAAM,SAAmB,mCAAmB;AAE5C,eAAO,GAAG,MAAM,QAAQ,MAAM,CAAC,UAAiB;AAC9C,gBAAM;AAAA;AAAA,UAGT,KAAK,MAAM;AACV,YAAG,MAAM,QAAQ;AAEf,iBAAO,UAAU,IAAI,OAAO,QAAQ,KAAK,MAAM;AAAA;AAEjD,eAAO;AAAA;AAAA,eAEH,MAAM,QAAQ;AAEtB,aAAO,UAAU,IAAI,OAAO,QAAQ,KAAK,MAAM;AAAA;AAEjD,WAAO;AAAA,KAER,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA;AAAA;AAIL,MAAM,eAAe,CAAC,WAA2B,OAAO,KAAK,QAAQ,SAAS;AAE9E,MAAM,eAAe,CAAC,eAA+B;AAC1D,QAAM,UAAU,CAAC,QAAgB,IAAI,MAAM,yCAAyC;AAEpF,MAAI,UAAU,QAAQ;AAEtB,MAAG,QAAQ,WAAW,GAAG;AAEvB,cAAU,QAAQ,mBAAmB;AAGrC,QAAG,QAAQ,WAAW,GAAG;AACvB,YAAM,MAAM;AAAA;AAAA;AAIhB,SAAO;AAAA,IACL,MAAM,OAAO,KAAK,QAAQ,IAAI;AAAA,IAC9B,MAAM,QAAQ;AAAA;AAAA;",
  "names": []
}

package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@nlabs/reaktor",
3
- "version": "0.5.3",
3
+ "version": "0.5.4",
4
4
  "description": "Reaktor",
5
5
  "main": "./lib/index.js",
6
6
  "module": "./lib/index.js",
@@ -61,8 +61,8 @@
61
61
  "net": "^1.0.2",
62
62
  "node-yelp": "^0.0.2",
63
63
  "numeral": "^2.0.6",
64
- "request-promise": "^4.2.6",
65
64
  "sanitize-html": "^2.5.1",
65
+ "spawn-sync": "^2.0.0",
66
66
  "stripe": "^8.176.0",
67
67
  "tls": "0.0.1",
68
68
  "to": "^0.2.9",