Endpoint.js

  1. import React, {DeviceEventEmitter, NativeModules} from 'react-native';
  2. import {EventEmitter} from 'events'
  3. import Call from './Call'
  4. import Message from './Message'
  5. import Account from './Account'
  6. /**
  7. * SIP headers object, where each key is a header name and value is a header value.
  8. * Example:
  9. * {
  10. * "X-Custom-Header": "Test Header Value",
  11. * "X-Custom-ID": "Awesome Header"
  12. * }
  13. *
  14. * @typedef {Object} PjSipHdrList
  15. */
  16. /**
  17. * An additional information to be sent with outgoing SIP message.
  18. * It can (optionally) be specified for example
  19. * with #Endpoint.makeCall(), #Endpoint.answerCall(), #Endpoint.hangupCall(),
  20. * #Endpoint.holdCall() and many more.
  21. *
  22. * @typedef {Object} PjSipMsgData
  23. * @property {String} target_uri - Indicates whether the Courage component is present.
  24. * @property {PjSipHdrList} hdr_list - Additional message headers as linked list.
  25. * @property {String} content_type - MIME type of optional message body.
  26. * @property {String} msg_body - MIME type of optional message body.
  27. */
  28. /**
  29. * An additional information to be sent with outgoing SIP message.
  30. * It can (optionally) be specified for example
  31. * with #Endpoint.makeCall(), #Endpoint.answerCall(), #Endpoint.hangupCall(),
  32. * #Endpoint.holdCall() and many more.
  33. *
  34. * @typedef {Object} PjSipCallSetttings
  35. * @property {number} flag - Bitmask of #pjsua_call_flag constants.
  36. * @property {number} req_keyframe_method - This flag controls what methods to request keyframe are allowed on the call.
  37. * @property {number} aud_cnt - Number of simultaneous active audio streams for this call. Setting this to zero will disable audio in this call.
  38. * @property {number} vid_cnt - Number of simultaneous active video streams for this call. Setting this to zero will disable video in this call.
  39. */
  40. export default class Endpoint extends EventEmitter {
  41. constructor() {
  42. super();
  43. // Subscribe to Accounts events
  44. DeviceEventEmitter.addListener('pjSipRegistrationChanged', this._onRegistrationChanged.bind(this));
  45. // Subscribe to Calls events
  46. DeviceEventEmitter.addListener('pjSipCallReceived', this._onCallReceived.bind(this));
  47. DeviceEventEmitter.addListener('pjSipCallChanged', this._onCallChanged.bind(this));
  48. DeviceEventEmitter.addListener('pjSipCallTerminated', this._onCallTerminated.bind(this));
  49. DeviceEventEmitter.addListener('pjSipCallScreenLocked', this._onCallScreenLocked.bind(this));
  50. DeviceEventEmitter.addListener('pjSipMessageReceived', this._onMessageReceived.bind(this));
  51. DeviceEventEmitter.addListener('pjSipConnectivityChanged', this._onConnectivityChanged.bind(this));
  52. }
  53. /**
  54. * Returns a Promise that will be resolved once PjSip module is initialized.
  55. * Do not call any function while library is not initialized.
  56. *
  57. * @returns {Promise}
  58. */
  59. start(configuration) {
  60. return new Promise(function(resolve, reject) {
  61. NativeModules.PjSipModule.start(configuration, (successful, data) => {
  62. if (successful) {
  63. let accounts = [];
  64. let calls = [];
  65. if (data.hasOwnProperty('accounts')) {
  66. for (let d of data['accounts']) {
  67. accounts.push(new Account(d));
  68. }
  69. }
  70. if (data.hasOwnProperty('calls')) {
  71. for (let d of data['calls']) {
  72. calls.push(new Call(d));
  73. }
  74. }
  75. let extra = {};
  76. for (let key in data) {
  77. if (data.hasOwnProperty(key) && key != "accounts" && key != "calls") {
  78. extra[key] = data[key];
  79. }
  80. }
  81. resolve({
  82. accounts,
  83. calls,
  84. ...extra
  85. });
  86. } else {
  87. reject(data);
  88. }
  89. });
  90. });
  91. }
  92. updateStunServers(accountId, stunServerList) {
  93. return new Promise(function(resolve, reject) {
  94. NativeModules.PjSipModule.updateStunServers(accountId, stunServerList, (successful, data) => {
  95. if (successful) {
  96. resolve(data);
  97. } else {
  98. reject(data);
  99. }
  100. })
  101. })
  102. }
  103. /**
  104. * @param configuration
  105. * @returns {Promise}
  106. */
  107. changeNetworkConfiguration(configuration) {
  108. return new Promise(function(resolve, reject) {
  109. NativeModules.PjSipModule.changeNetworkConfiguration(configuration, (successful, data) => {
  110. if (successful) {
  111. resolve(data);
  112. } else {
  113. reject(data);
  114. }
  115. });
  116. });
  117. }
  118. /**
  119. * @param configuration
  120. * @returns {Promise}
  121. */
  122. changeServiceConfiguration(configuration) {
  123. return new Promise(function(resolve, reject) {
  124. NativeModules.PjSipModule.changeServiceConfiguration(configuration, (successful, data) => {
  125. if (successful) {
  126. resolve(data);
  127. } else {
  128. reject(data);
  129. }
  130. });
  131. });
  132. }
  133. /**
  134. * Add a new account. If registration is configured for this account, this function would also start the
  135. * SIP registration session with the SIP registrar server. This SIP registration session will be maintained
  136. * internally by the library, and application doesn't need to do anything to maintain the registration session.
  137. *
  138. * An example configuration:
  139. * {
  140. * name: "John Doe",
  141. * username: "100",
  142. * domain: "pbx.com",
  143. * password: "XXXXXX",
  144. *
  145. * proxy: "192.168.100.1:5060", // default disabled.
  146. * transport: "TCP", // default TCP
  147. * regServer: "pbx.com", // default taken from domain
  148. * regTimeout: 300, // default 300
  149. * }
  150. *
  151. * @param {Object} configuration
  152. * @returns {Promise}
  153. */
  154. createAccount(configuration) {
  155. return new Promise(function(resolve, reject) {
  156. NativeModules.PjSipModule.createAccount(configuration, (successful, data) => {
  157. if (successful) {
  158. resolve(new Account(data));
  159. } else {
  160. reject(data);
  161. }
  162. });
  163. });
  164. }
  165. replaceAccount(account, configuration) {
  166. throw new Error("Not implemented");
  167. }
  168. /**
  169. * Update registration or perform unregistration.
  170. * If registration is configured for this account, then initial SIP REGISTER will be sent when the account is added.
  171. * Application normally only need to call this function if it wants to manually update the registration or to unregister from the server.
  172. *
  173. * @param {Account} account
  174. * @param bool renew If renew argument is zero, this will start unregistration process.
  175. * @returns {Promise}
  176. */
  177. registerAccount(account, renew = true) {
  178. return new Promise(function(resolve, reject) {
  179. NativeModules.PjSipModule.registerAccount(account.getId(), renew, (successful, data) => {
  180. if (successful) {
  181. resolve(data);
  182. } else {
  183. reject(data);
  184. }
  185. });
  186. });
  187. }
  188. /**
  189. * Delete an account. This will unregister the account from the SIP server, if necessary, and terminate server side presence subscriptions associated with this account.
  190. *
  191. * @param {Account} account
  192. * @returns {Promise}
  193. */
  194. deleteAccount(account) {
  195. return new Promise(function(resolve, reject) {
  196. NativeModules.PjSipModule.deleteAccount(account.getId(), (successful, data) => {
  197. if (successful) {
  198. resolve(data);
  199. } else {
  200. reject(data);
  201. }
  202. });
  203. });
  204. }
  205. /**
  206. * Make an outgoing call to the specified URI.
  207. * Available call settings:
  208. * - audioCount - Number of simultaneous active audio streams for this call. Setting this to zero will disable audio in this call.
  209. * - videoCount - Number of simultaneous active video streams for this call. Setting this to zero will disable video in this call.
  210. * -
  211. *
  212. * @param account {Account}
  213. * @param destination {String} Destination SIP URI.
  214. * @param callSettings {PjSipCallSetttings} Outgoing call settings.
  215. * @param msgSettings {PjSipMsgData} Outgoing call additional information to be sent with outgoing SIP message.
  216. */
  217. makeCall(account, destination, callSettings, msgData) {
  218. destination = this._normalize(account, destination);
  219. return new Promise(function(resolve, reject) {
  220. NativeModules.PjSipModule.makeCall(account.getId(), destination, callSettings, msgData, (successful, data) => {
  221. if (successful) {
  222. resolve(new Call(data));
  223. } else {
  224. reject(data);
  225. }
  226. });
  227. });
  228. }
  229. /**
  230. * Send response to incoming INVITE request.
  231. *
  232. * @param call {Call} Call instance
  233. * @returns {Promise}
  234. */
  235. answerCall(call) {
  236. return new Promise((resolve, reject) => {
  237. NativeModules.PjSipModule.answerCall(call.getId(), (successful, data) => {
  238. if (successful) {
  239. resolve(data);
  240. } else {
  241. reject(data);
  242. }
  243. });
  244. });
  245. }
  246. /**
  247. * Hangup call by using method that is appropriate according to the call state.
  248. *
  249. * @param call {Call} Call instance
  250. * @returns {Promise}
  251. */
  252. hangupCall(call) {
  253. // TODO: Add possibility to pass code and reason for hangup.
  254. return new Promise((resolve, reject) => {
  255. NativeModules.PjSipModule.hangupCall(call.getId(), (successful, data) => {
  256. if (successful) {
  257. resolve(data);
  258. } else {
  259. reject(data);
  260. }
  261. });
  262. });
  263. }
  264. /**
  265. * Hangup call by using Decline (603) method.
  266. *
  267. * @param call {Call} Call instance
  268. * @returns {Promise}
  269. */
  270. declineCall(call) {
  271. return new Promise((resolve, reject) => {
  272. NativeModules.PjSipModule.declineCall(call.getId(), (successful, data) => {
  273. if (successful) {
  274. resolve(data);
  275. } else {
  276. reject(data);
  277. }
  278. });
  279. });
  280. }
  281. /**
  282. * Put the specified call on hold. This will send re-INVITE with the appropriate SDP to inform remote that the call is being put on hold.
  283. *
  284. * @param call {Call} Call instance
  285. * @returns {Promise}
  286. */
  287. holdCall(call) {
  288. return new Promise((resolve, reject) => {
  289. NativeModules.PjSipModule.holdCall(call.getId(), (successful, data) => {
  290. if (successful) {
  291. resolve(data);
  292. } else {
  293. reject(data);
  294. }
  295. });
  296. });
  297. }
  298. /**
  299. * Release the specified call from hold. This will send re-INVITE with the appropriate SDP to inform remote that the call is resumed.
  300. *
  301. * @param call {Call} Call instance
  302. * @returns {Promise}
  303. */
  304. unholdCall(call) {
  305. return new Promise((resolve, reject) => {
  306. NativeModules.PjSipModule.unholdCall(call.getId(), (successful, data) => {
  307. if (successful) {
  308. resolve(data);
  309. } else {
  310. reject(data);
  311. }
  312. });
  313. });
  314. }
  315. /**
  316. * @param call {Call} Call instance
  317. * @returns {Promise}
  318. */
  319. muteCall(call) {
  320. return new Promise((resolve, reject) => {
  321. NativeModules.PjSipModule.muteCall(call.getId(), (successful, data) => {
  322. if (successful) {
  323. resolve(data);
  324. } else {
  325. reject(data);
  326. }
  327. });
  328. });
  329. }
  330. /**
  331. * @param call {Call} Call instance
  332. * @returns {Promise}
  333. */
  334. unMuteCall(call) {
  335. return new Promise((resolve, reject) => {
  336. NativeModules.PjSipModule.unMuteCall(call.getId(), (successful, data) => {
  337. if (successful) {
  338. resolve(data);
  339. } else {
  340. reject(data);
  341. }
  342. });
  343. });
  344. }
  345. /**
  346. * @param call {Call} Call instance
  347. * @returns {Promise}
  348. */
  349. useSpeaker(call) {
  350. return new Promise((resolve, reject) => {
  351. NativeModules.PjSipModule.useSpeaker(call.getId(), (successful, data) => {
  352. if (successful) {
  353. resolve(data);
  354. } else {
  355. reject(data);
  356. }
  357. });
  358. });
  359. }
  360. /**
  361. * @param call {Call} Call instance
  362. * @returns {Promise}
  363. */
  364. useEarpiece(call) {
  365. return new Promise((resolve, reject) => {
  366. NativeModules.PjSipModule.useEarpiece(call.getId(), (successful, data) => {
  367. if (successful) {
  368. resolve(data);
  369. } else {
  370. reject(data);
  371. }
  372. });
  373. });
  374. }
  375. /**
  376. * Initiate call transfer to the specified address.
  377. * This function will send REFER request to instruct remote call party to initiate a new INVITE session to the specified destination/target.
  378. *
  379. * @param account {Account} Account associated with call.
  380. * @param call {Call} The call to be transferred.
  381. * @param destination URI of new target to be contacted. The URI may be in name address or addr-spec format.
  382. * @returns {Promise}
  383. */
  384. xferCall(account, call, destination) {
  385. destination = this._normalize(account, destination);
  386. return new Promise((resolve, reject) => {
  387. NativeModules.PjSipModule.xferCall(call.getId(), destination, (successful, data) => {
  388. if (successful) {
  389. resolve(data);
  390. } else {
  391. reject(data);
  392. }
  393. });
  394. });
  395. }
  396. /**
  397. * Initiate attended call transfer.
  398. * This function will send REFER request to instruct remote call party to initiate new INVITE session to the URL of destCall.
  399. * The party at destCall then should "replace" the call with us with the new call from the REFER recipient.
  400. *
  401. * @param call {Call} The call to be transferred.
  402. * @param destCall {Call} The call to be transferred.
  403. * @returns {Promise}
  404. */
  405. xferReplacesCall(call, destCall) {
  406. return new Promise((resolve, reject) => {
  407. NativeModules.PjSipModule.xferReplacesCall(call.getId(), destCall.getId(), (successful, data) => {
  408. if (successful) {
  409. resolve(data);
  410. } else {
  411. reject(data);
  412. }
  413. });
  414. });
  415. }
  416. /**
  417. * Redirect (forward) specified call to destination.
  418. * This function will send response to INVITE to instruct remote call party to redirect incoming call to the specified destination/target.
  419. *
  420. * @param account {Account} Account associated with call.
  421. * @param call {Call} The call to be transferred.
  422. * @param destination URI of new target to be contacted. The URI may be in name address or addr-spec format.
  423. * @returns {Promise}
  424. */
  425. redirectCall(account, call, destination) {
  426. destination = this._normalize(account, destination);
  427. return new Promise((resolve, reject) => {
  428. NativeModules.PjSipModule.redirectCall(call.getId(), destination, (successful, data) => {
  429. if (successful) {
  430. resolve(data);
  431. } else {
  432. reject(data);
  433. }
  434. });
  435. });
  436. }
  437. /**
  438. * Send DTMF digits to remote using RFC 2833 payload formats.
  439. *
  440. * @param call {Call} Call instance
  441. * @param digits {String} DTMF string digits to be sent as described on RFC 2833 section 3.10.
  442. * @returns {Promise}
  443. */
  444. dtmfCall(call, digits) {
  445. return new Promise((resolve, reject) => {
  446. NativeModules.PjSipModule.dtmfCall(call.getId(), digits, (successful, data) => {
  447. if (successful) {
  448. resolve(data);
  449. } else {
  450. reject(data);
  451. }
  452. });
  453. });
  454. }
  455. activateAudioSession() {
  456. return new Promise((resolve, reject) => {
  457. NativeModules.PjSipModule.activateAudioSession((successful, data) => {
  458. if (successful) {
  459. resolve(data);
  460. } else {
  461. reject(data);
  462. }
  463. });
  464. });
  465. }
  466. deactivateAudioSession() {
  467. return new Promise((resolve, reject) => {
  468. NativeModules.PjSipModule.deactivateAudioSession((successful, data) => {
  469. if (successful) {
  470. resolve(data);
  471. } else {
  472. reject(data);
  473. }
  474. });
  475. });
  476. }
  477. changeOrientation(orientation) {
  478. const orientations = [
  479. 'PJMEDIA_ORIENT_UNKNOWN',
  480. 'PJMEDIA_ORIENT_ROTATE_90DEG',
  481. 'PJMEDIA_ORIENT_ROTATE_270DEG',
  482. 'PJMEDIA_ORIENT_ROTATE_180DEG',
  483. 'PJMEDIA_ORIENT_NATURAL'
  484. ]
  485. if (orientations.indexOf(orientation) === -1) {
  486. throw new Error(`Invalid ${JSON.stringify(orientation)} device orientation, but expected ${orientations.join(", ")} values`)
  487. }
  488. NativeModules.PjSipModule.changeOrientation(orientation)
  489. }
  490. changeCodecSettings(codecSettings) {
  491. return new Promise(function(resolve, reject) {
  492. NativeModules.PjSipModule.changeCodecSettings(codecSettings, (successful, data) => {
  493. if (successful) {
  494. resolve(data);
  495. } else {
  496. reject(data);
  497. }
  498. });
  499. });
  500. }
  501. /**
  502. * @fires Endpoint#connectivity_changed
  503. * @private
  504. * @param data {Object}
  505. */
  506. _onConnectivityChanged(data) {
  507. /**
  508. * Fires when registration status has changed.
  509. *
  510. * @event Endpoint#connectivity_changed
  511. * @property {Account} account
  512. */
  513. this.emit("connectivity_changed", new Account(data));
  514. }
  515. /**
  516. * @fires Endpoint#registration_changed
  517. * @private
  518. * @param data {Object}
  519. */
  520. _onRegistrationChanged(data) {
  521. /**
  522. * Fires when registration status has changed.
  523. *
  524. * @event Endpoint#registration_changed
  525. * @property {Account} account
  526. */
  527. this.emit("registration_changed", new Account(data));
  528. }
  529. /**
  530. * @fires Endpoint#call_received
  531. * @private
  532. * @param data {Object}
  533. */
  534. _onCallReceived(data) {
  535. /**
  536. * TODO
  537. *
  538. * @event Endpoint#call_received
  539. * @property {Call} call
  540. */
  541. this.emit("call_received", new Call(data));
  542. }
  543. /**
  544. * @fires Endpoint#call_changed
  545. * @private
  546. * @param data {Object}
  547. */
  548. _onCallChanged(data) {
  549. /**
  550. * TODO
  551. *
  552. * @event Endpoint#call_changed
  553. * @property {Call} call
  554. */
  555. this.emit("call_changed", new Call(data));
  556. }
  557. /**
  558. * @fires Endpoint#call_terminated
  559. * @private
  560. * @param data {Object}
  561. */
  562. _onCallTerminated(data) {
  563. /**
  564. * TODO
  565. *
  566. * @event Endpoint#call_terminated
  567. * @property {Call} call
  568. */
  569. this.emit("call_terminated", new Call(data));
  570. }
  571. /**
  572. * @fires Endpoint#call_screen_locked
  573. * @private
  574. * @param lock bool
  575. */
  576. _onCallScreenLocked(lock) {
  577. /**
  578. * TODO
  579. *
  580. * @event Endpoint#call_screen_locked
  581. * @property bool lock
  582. */
  583. this.emit("call_screen_locked", lock);
  584. }
  585. /**
  586. * @fires Endpoint#message_received
  587. * @private
  588. * @param data {Object}
  589. */
  590. _onMessageReceived(data) {
  591. /**
  592. * TODO
  593. *
  594. * @event Endpoint#message_received
  595. * @property {Message} message
  596. */
  597. this.emit("message_received", new Message(data));
  598. }
  599. /**
  600. * @fires Endpoint#connectivity_changed
  601. * @private
  602. * @param available bool
  603. */
  604. _onConnectivityChanged(available) {
  605. /**
  606. * @event Endpoint#connectivity_changed
  607. * @property bool available True if connectivity matches current Network settings, otherwise false.
  608. */
  609. this.emit("connectivity_changed", available);
  610. }
  611. /**
  612. * Normalize Destination URI
  613. *
  614. * @param account
  615. * @param destination {string}
  616. * @returns {string}
  617. * @private
  618. */
  619. _normalize(account, destination) {
  620. if (!destination.startsWith("sip:")) {
  621. let realm = account.getRegServer();
  622. if (!realm) {
  623. realm = account.getDomain();
  624. let s = realm.indexOf(":");
  625. if (s > 0) {
  626. realm = realm.substr(0, s + 1);
  627. }
  628. }
  629. destination = "sip:" + destination + "@" + realm;
  630. }
  631. return destination;
  632. }
  633. // setUaConfig(UaConfig value)
  634. // setMaxCalls
  635. // setUserAgent
  636. // setNatTypeInSdp
  637. // setLogConfig(LogConfig value)
  638. // setLevel
  639. }