Source mondo.js

  1. ;(function () {
  2. 'use strict'
  3. var request = require('request')
  4. var _ = require('lodash')
  5. var version = require('../package.json').version
  6. var apiValues = require('./api.values.json')
  7. var apiHost = apiValues.host
  8. var methodPaths = apiValues.resources
  9. var dateParams = ['since', 'before']
  10. var client_id
  11. var client_secret
  12. /**
  13. * @module mondo-bank
  14. * @description
  15. * ## Usage
  16. *
  17. * mondo = require('mondo-bank')
  18. *
  19. * All methods return a promise but can optionally be called with a callback function as the final argument
  20. *
  21. * #### Promise style
  22. *
  23. * methodPromise = mondo[$method]([$params])
  24. * methodPromise
  25. * .then(function(value){
  26. * ...
  27. * })
  28. * .catch(function(err){
  29. * ...
  30. * })
  31. *
  32. * #### Callback style
  33. *
  34. * mondo[method]([$params], function(err, value){
  35. * if (err) {
  36. * ...
  37. * }
  38. * ...
  39. * })
  40. */
  41. // Helper functions
  42. // Allow params to be aliased
  43. function dealiasParams (params, aliases) {
  44. params = _.extend({}, params)
  45. Object.keys(aliases).forEach(function (param) {
  46. var aliased = aliases[param]
  47. aliased = typeof aliased === 'string' ? [aliased] : aliased
  48. aliased.forEach(function (alias) {
  49. if (params[param] === undefined && params[alias] !== undefined) {
  50. params[param] = params[alias]
  51. }
  52. delete params[alias]
  53. })
  54. })
  55. return params
  56. }
  57. // Allow params to be passed unbracketed
  58. function bracketifyParams (obj, param) {
  59. var bracketedObj = {}
  60. Object.keys(obj).forEach(function (prop) {
  61. // what if bracketed prop already exists?
  62. bracketedObj[param + '[' + prop + ']'] = obj[prop]
  63. })
  64. return bracketedObj
  65. }
  66. // Adds necessary auth header
  67. function addAuthorizationHeader (options, access_token) {
  68. options.headers = options.headers || {}
  69. options.headers.Authorization = 'Bearer ' + access_token
  70. return options
  71. }
  72. function handleApiResponse (options, resolve, reject) {
  73. request(options, function (err, res) {
  74. if (err) {
  75. reject(err)
  76. } else {
  77. var data = res.body
  78. if (res.statusCode === 200) {
  79. resolve(data)
  80. } else {
  81. reject({
  82. status: res.statusCode,
  83. error: data
  84. })
  85. }
  86. }
  87. })
  88. }
  89. // Call the API
  90. function apiRequest (options, fn) {
  91. options = _.extend({}, options)
  92. if (options.form && options.form.client_id && options.form.client_secret) {
  93. client_id = options.form.client_id
  94. client_secret = options.form.client_secret
  95. }
  96. if (options.qs) {
  97. dateParams.forEach(function (dParam) {
  98. if (typeof options.qs[dParam] === 'object') {
  99. options.qs[dParam] = options.qs[dParam].toISOString()
  100. }
  101. })
  102. }
  103. options.uri = apiHost + options.uri
  104. options.json = true
  105. options.headers = options.headers || {}
  106. options.headers.client = 'NodeMondo-v' + version
  107. options.method = options.method || 'GET'
  108. if (typeof Promise === 'undefined') {
  109. if (!fn) {
  110. var noCallbackError = new Error('method.missing.callback')
  111. throw noCallbackError
  112. }
  113. return handleApiResponse(options, function (res) {
  114. fn(null, res)
  115. }, fn)
  116. }
  117. var reqpromise = new Promise(function (resolve, reject) {
  118. handleApiResponse(options, resolve, reject)
  119. })
  120. if (!fn) {
  121. return reqpromise
  122. } else {
  123. reqpromise.then(function (parsed_body) {
  124. fn(null, parsed_body)
  125. })
  126. .catch(function (err) {
  127. fn(err)
  128. })
  129. }
  130. }
  131. // Call the API with authentication
  132. function apiRequestAuthenticated (options, access_token, fn) {
  133. options = _.extend({}, options)
  134. options = addAuthorizationHeader(options, access_token)
  135. return apiRequest(options, fn)
  136. }
  137. // API methods
  138. /**
  139. * @method token
  140. * @static
  141. * @param {object} credentials Mondo credentials
  142. * @param {string} credentials.client_id Dev API client id
  143. * @param {string} credentials.client_secret Dev API client secret
  144. * @param {string} credentials.username Mondo user’s username
  145. * @param {string} credentials.password Mondo user’s password
  146. * @param {function} [fn] Callback function
  147. * @return {object} Response value
  148. * @description
  149. * Acquire an access token
  150. *
  151. * tokenPromise = mondo.token({
  152. * client_id: client_id,
  153. * client_secret: client_secret,
  154. * username: username,
  155. * password: password
  156. * })
  157. *
  158. * @see https://getmondo.co.uk/docs/#acquiring-an-access-token
  159. */
  160. function token (credentials, fn) {
  161. var options = {
  162. method: 'POST',
  163. uri: methodPaths.token,
  164. form: _.extend({
  165. grant_type: 'password',
  166. client_id: client_id,
  167. client_secret: client_secret
  168. }, credentials)
  169. }
  170. return apiRequest(options, fn)
  171. }
  172. /**
  173. * @method tokenInfo
  174. * @static
  175. * @param {string} access_token Access token
  176. * @param {function} [fn] Callback function
  177. * @return {object} Response value
  178. * @description
  179. * Get information about an access token
  180. *
  181. * tokenInfoPromise = mondo.tokenInfo(accessToken)
  182. *
  183. * @see https://getmondo.co.uk/docs/#authenticating-requests
  184. */
  185. function tokenInfo (access_token, fn) {
  186. var options = {
  187. uri: methodPaths.tokenInfo
  188. }
  189. return apiRequestAuthenticated(options, access_token, fn)
  190. }
  191. /**
  192. * @method refreshToken
  193. * @static
  194. * @param {string} refresh_token Refresh token
  195. * @param {function} [fn] Callback function
  196. * @return {object} Response value
  197. * @description
  198. * Refresh a proviously acquired token
  199. *
  200. * refreshTokenPromise = mondo.refreshToken(refreshToken)
  201. *
  202. * or if the client id and secret have not been previously passed
  203. *
  204. * refreshTokenPromise = mondo.refreshToken({
  205. * refreshToken: refreshToken,
  206. * client_id: client_id,
  207. * client_secret: client_secret
  208. * })
  209. *
  210. * @see https://getmondo.co.uk/docs/#refreshing-access
  211. */
  212. function refreshToken (refresh_token, fn) {
  213. if (typeof refresh_token === 'string') {
  214. refresh_token = {
  215. refresh_token: refresh_token
  216. }
  217. }
  218. var options = {
  219. method: 'POST',
  220. uri: methodPaths.refreshToken,
  221. form: _.extend({
  222. grant_type: 'refresh_token',
  223. client_id: client_id,
  224. client_secret: client_secret
  225. }, refresh_token)
  226. }
  227. return apiRequest(options, fn)
  228. }
  229. /**
  230. * @method accounts
  231. * @static
  232. * @param {string} access_token Access token
  233. * @param {function} [fn] Callback function
  234. * @return {object} Response value
  235. * @description
  236. * Get detailed information about customer’s accounts
  237. *
  238. * accountsPromise = mondo.accounts(accessToken)
  239. *
  240. * @see https://getmondo.co.uk/docs/#list-accounts
  241. */
  242. function accounts (access_token, fn) {
  243. var options = {
  244. uri: methodPaths.accounts
  245. }
  246. return apiRequestAuthenticated(options, access_token, fn)
  247. }
  248. /**
  249. * @method balance
  250. * @static
  251. * @param {string} account_id Account id
  252. * @param {string} access_token Access token
  253. * @param {function} [fn] Callback function
  254. * @return {object} Response value
  255. * @description
  256. * Get balance details for an account
  257. *
  258. * balancePromise = mondo.balance(account_id, access_token)
  259. *
  260. * @see https://getmondo.co.uk/docs/#read-balance
  261. */
  262. function balance (account_id, access_token, fn) {
  263. var options = {
  264. uri: methodPaths.balance,
  265. qs: {
  266. account_id: account_id
  267. }
  268. }
  269. return apiRequestAuthenticated(options, access_token, fn)
  270. }
  271. /**
  272. * @method transactions
  273. * @static
  274. * @param {object|string} params Query object
  275. * If passed as a string, used as account_id
  276. * @param {string} params.account_id Account id
  277. * @param {date|string} [params.since] Date after which to show results (Date object or ISO string)
  278. * @param {date|string} [params.before] Date before which to show results (Date object or ISO string)
  279. * @param {int} [params.limit] Max number of transactions to return (100 max)
  280. * @param {string} access_token Access token
  281. * @param {function} [fn] Callback function
  282. * @return {object} Response value
  283. * @description
  284. * List transactions
  285. *
  286. * transactionsPromise = mondo.transactions(account_id, access_token)
  287. *
  288. * or to filter the results
  289. *
  290. * transactionsPromise = mondo.transactions({
  291. * account_id: account_id,
  292. * since: since,
  293. * before: before
  294. * limit: limit
  295. * }, access_token)
  296. *
  297. * @see https://getmondo.co.uk/docs/#list-transactions
  298. */
  299. function transactions (params, access_token, fn) {
  300. if (typeof params === 'string') {
  301. params = {
  302. account_id: params
  303. }
  304. }
  305. var options = {
  306. uri: methodPaths.transactions,
  307. qs: params
  308. }
  309. return apiRequestAuthenticated(options, access_token, fn)
  310. }
  311. /**
  312. * @method transaction
  313. * @static
  314. * @param {object|String} params Transaction params
  315. * @param {string} params.transaction_id Transaction ID
  316. * @param {string} [params.expand] Property to expand (merchant)
  317. * @param {string} access_token Access token
  318. * @param {function} [fn] Callback function
  319. * @return {object} Response value
  320. * @description
  321. * Get details about a transaction
  322. *
  323. * transactionPromise = mondo.transaction(transaction_id, access_token)
  324. *
  325. * or to see expanded info for the merchant
  326. *
  327. * transactionPromise = mondo.transaction({
  328. * transaction_id: transaction_id,
  329. * expand: 'merchant'
  330. * }, access_token)
  331. *
  332. * @see https://getmondo.co.uk/docs/#retrieve-transaction
  333. */
  334. function transaction (params, access_token, fn) {
  335. var transaction_id
  336. if (typeof params === 'string') {
  337. transaction_id = params
  338. params = undefined
  339. } else {
  340. transaction_id = params.transaction_id
  341. delete params.transaction_id
  342. }
  343. if (params && params.expand) {
  344. params['expand[]'] = params.expand
  345. delete params.expand
  346. }
  347. var options = {
  348. uri: methodPaths.transaction + transaction_id,
  349. qs: params
  350. }
  351. return apiRequestAuthenticated(options, access_token, fn)
  352. }
  353. /**
  354. * @method annotateTransaction
  355. * @static
  356. * @param {string|object} transaction_id Transaction ID
  357. * @param {string} [transaction_id.transaction_id] Transaction ID, if passed as object
  358. * @param {string} [transaction_id.metadata] Metadata, if passed as nested object
  359. * @param {object} metadata Annotation metadata object
  360. * @param {object} annotation Alias for metadata
  361. * @param {object} annotations Alias for metadata
  362. * @param {string} access_token Access token
  363. * @param {function} [fn] Callback function
  364. * @return {object} Response value
  365. * @description
  366. * Annotate a transaction
  367. *
  368. * annotateTransactionPromise = mondo.annotateTransaction(transaction_id, {
  369. * foo: 'bar'
  370. * }, access_token)
  371. *
  372. * or
  373. *
  374. * annotateTransactionPromise = mondo.annotateTransaction({
  375. * transaction_id: transaction_id,
  376. * foo: 'bar'
  377. * }, access_token)
  378. *
  379. * or
  380. *
  381. * annotateTransactionPromise = mondo.annotateTransaction({
  382. * transaction_id: transaction_id,
  383. * metadata: {
  384. * foo: 'bar'
  385. * }
  386. * }, access_token)
  387. *
  388. * @see https://getmondo.co.uk/docs/#annotate-transaction
  389. */
  390. function annotateTransaction (transaction_id, metadata, access_token, fn) {
  391. if (typeof transaction_id === 'object') {
  392. fn = access_token
  393. access_token = metadata
  394. metadata = _.extend({}, transaction_id)
  395. transaction_id = metadata.transaction_id
  396. delete metadata.transaction_id
  397. metadata = dealiasParams(metadata, {
  398. metadata: 'annotation'
  399. })
  400. metadata = metadata.metadata || metadata
  401. }
  402. metadata = bracketifyParams(metadata, 'metadata')
  403. var options = {
  404. method: 'PATCH',
  405. uri: methodPaths.annotateTransaction + transaction_id,
  406. form: metadata
  407. }
  408. return apiRequestAuthenticated(options, access_token, fn)
  409. }
  410. /**
  411. * @method createFeedItem
  412. * @static
  413. * @param {object} params Params object
  414. * @param {string} params.account_id Account ID
  415. * @param {object} params.params Feed item params
  416. * @param {string} params.params.title Title for feed item
  417. * @param {string} params.params.image_url Icon url to use as icon
  418. * @param {string} [params.params.body] Body text to display
  419. * @param {string} [params.params.background_color] Background colour
  420. * @param {string} [params.params.title_color] Title colour
  421. * @param {string} [params.params.body_color] Body colour
  422. * @param {string} params.url Feed item url
  423. * @param {string} access_token Access token
  424. * @param {function} [fn] Callback function
  425. * @return {object} Response value
  426. * @description
  427. * Publish a new feed entry
  428. *
  429. * createFeedItemPromise = mondo.createFeedItem({
  430. * account_id: accountId,
  431. * params: {
  432. * title: title,
  433. * image_url: image_url
  434. * },
  435. * url: url
  436. * }, access_token)
  437. *
  438. * @see https://getmondo.co.uk/docs/#create-feed-item
  439. */
  440. function createFeedItem (params, access_token, fn) {
  441. params = _.extend({}, params)
  442. params.type = params.type || 'basic'
  443. var feedParams = bracketifyParams(params.params, 'params')
  444. delete params.params
  445. params = _.extend(params, feedParams)
  446. var options = {
  447. method: 'POST',
  448. uri: methodPaths.createFeedItem,
  449. form: params
  450. }
  451. return apiRequestAuthenticated(options, access_token, fn)
  452. }
  453. /**
  454. * @method registerWebhook
  455. * @static
  456. * @param {string} account_id Account ID
  457. * @param {string} url Webhook url
  458. * @param {string} access_token Access token
  459. * @param {function} [fn] Callback function
  460. * @return {object} Response value
  461. * @description
  462. * Register a webhook
  463. *
  464. * registerWebhookPromise = mondo.registerWebhook(account_id, url, access_token)
  465. *
  466. * See https://getmondo.co.uk/docs/#transaction-created for details of the transaction.created event which is sent to the webhook each time a new transaction is created in a user’s account
  467. *
  468. * @see https://getmondo.co.uk/docs/#registering-a-web-hook
  469. */
  470. function registerWebhook (account_id, url, access_token, fn) {
  471. var options = {
  472. method: 'POST',
  473. uri: methodPaths.registerWebhook,
  474. form: {
  475. account_id: account_id,
  476. url: url
  477. }
  478. }
  479. return apiRequestAuthenticated(options, access_token, fn)
  480. }
  481. /**
  482. * @method webhooks
  483. * @static
  484. * @param {string} account_id Account ID
  485. * @param {string} access_token Access token
  486. * @param {function} [fn] Callback function
  487. * @return {object} Response value
  488. * @description
  489. * List webhooks
  490. *
  491. * webhooksPromise = mondo.webhooks(account_id, access_token)
  492. *
  493. * @see https://getmondo.co.uk/docs/#list-web-hooks
  494. */
  495. function webhooks (account_id, access_token, fn) {
  496. var options = {
  497. uri: methodPaths.webhooks,
  498. qs: {
  499. account_id: account_id
  500. }
  501. }
  502. return apiRequestAuthenticated(options, access_token, fn)
  503. }
  504. /**
  505. * @method deleteWebhook
  506. * @static
  507. * @param {string} webhook_id Webhook ID
  508. * @param {string} access_token Access token
  509. * @param {function} [fn] Callback function
  510. * @return {object} Response value
  511. * @description
  512. * Delete webhook
  513. *
  514. * deleteWebhookPromise = mondo.deleteWebhook(webhook_id, access_token)
  515. *
  516. * @see https://getmondo.co.uk/docs/#deleting-a-web-hook
  517. */
  518. function deleteWebhook (webhook_id, access_token, fn) {
  519. var options = {
  520. method: 'DELETE',
  521. uri: '/webhooks/' + webhook_id
  522. }
  523. return apiRequestAuthenticated(options, access_token, fn)
  524. }
  525. /**
  526. * @method registerAttachment
  527. * @static
  528. * @param {object} params Params object
  529. * @param {string} params.external_id Transaction ID
  530. * @param {string} params.file_type File type
  531. * @param {string} params.file_url File url
  532. * @param {string} [params.transaction_id] Alias for external_id
  533. * @param {string} [params.id] Alias for external_id
  534. * @param {string} [params.type] Alias for file_type
  535. * @param {string} [params.url] Alias for file_url
  536. * @param {string} access_token Access token
  537. * @param {function} [fn] Callback function
  538. * @return {object} Response value
  539. * @description
  540. * Register attachment
  541. *
  542. * registerAttachmentPromise = mondo.registerAttachment({
  543. * external_id: transaction_id,
  544. * file_type: file_type,
  545. * file_url: file_url
  546. * }, access_token)
  547. *
  548. * @see https://getmondo.co.uk/docs/#register-attachment
  549. */
  550. function registerAttachment (params, access_token, fn) {
  551. params = dealiasParams(params, {
  552. external_id: ['transaction_id', 'id'],
  553. file_type: 'type',
  554. file_url: 'url'
  555. })
  556. var options = {
  557. method: 'POST',
  558. uri: methodPaths.registerAttachment,
  559. form: params
  560. }
  561. return apiRequestAuthenticated(options, access_token, fn)
  562. }
  563. /**
  564. * @method uploadAttachment
  565. * @static
  566. * @description
  567. * Request upload attachment url
  568. *
  569. * uploadAttachmentPromise = mondo.uploadAttachment({
  570. * file_name: file_name,
  571. * file_type: file_type
  572. * }, access_token)
  573. *
  574. * @see https://getmondo.co.uk/docs/#upload-attachment
  575. * @param {object} params params object
  576. * @param {string} params.file_name File name
  577. * @param {string} params.file_type File type
  578. * @param {string} [params.file] Alias for file_name
  579. * @param {string} [params.name] Alias for file_name
  580. * @param {string} [params.type] Alias for file_type
  581. * @param {string} access_token Access token
  582. * @param {function} [fn] Callback function
  583. * @return {object} Response value
  584. */
  585. function uploadAttachment (params, access_token, fn) {
  586. params = dealiasParams(params, {
  587. file_name: ['file', 'name'],
  588. file_type: 'type'
  589. })
  590. var options = {
  591. method: 'POST',
  592. uri: methodPaths.uploadAttachment,
  593. form: params
  594. }
  595. return apiRequestAuthenticated(options, access_token, fn)
  596. /*
  597. // rough on the train pretend code to allow upload and register in one call
  598. if (params.transaction_id) {
  599. // resolve apiRequest and attach image to it
  600. var upload = apiRequest(options)
  601. if (! fn) {
  602. upload.then(function(parsed_body) {
  603. fn(null, parsed_body)
  604. })
  605. .catch(function(err) {
  606. fn(err)
  607. })
  608. } else {
  609. return apiRequest(options, fn)
  610. }
  611. */
  612. }
  613. /**
  614. * @method deregisterAttachment
  615. * @static
  616. * @param {string} attachment_id Attachment ID
  617. * @param {string} access_token Access token
  618. * @param {function} [fn] Callback function
  619. * @return {object} Response value
  620. * @description
  621. * Deregister attachment
  622. *
  623. * deregisterAttachmentPromise = mondo.deregisterAttachment(attachment_id, access_token)
  624. *
  625. * @see https://getmondo.co.uk/docs/#deregister-attachment
  626. */
  627. function deregisterAttachment (attachment_id, access_token, fn) {
  628. var options = {
  629. method: 'POST',
  630. uri: methodPaths.deregisterAttachment,
  631. form: {
  632. id: attachment_id
  633. }
  634. }
  635. return apiRequestAuthenticated(options, access_token, fn)
  636. }
  637. module.exports = {
  638. token: token,
  639. tokenInfo: tokenInfo,
  640. refreshToken: refreshToken,
  641. accounts: accounts,
  642. balance: balance,
  643. transactions: transactions,
  644. transaction: transaction,
  645. annotateTransaction: annotateTransaction,
  646. createFeedItem: createFeedItem,
  647. webhooks: webhooks,
  648. registerWebhook: registerWebhook,
  649. deleteWebhook: deleteWebhook,
  650. uploadAttachment: uploadAttachment,
  651. registerAttachment: registerAttachment,
  652. deregisterAttachment: deregisterAttachment
  653. }
  654. })()