Source: models/users/matrix.js

  1. "use strict";
  2. /**
  3. * Construct a Matrix user.
  4. * @constructor
  5. * @param {string} userId The user_id of the user.
  6. * @param {Object=} data Serialized data values
  7. * @param {boolean} escape [true] Escape the user's localpart. Modify {@link MatrixUser~ESCAPE_DEFAULT}
  8. * to change the default value.
  9. */
  10. function MatrixUser(userId, data, escape=MatrixUser.ESCAPE_DEFAULT) {
  11. if (!userId) {
  12. throw new Error("Missing user_id");
  13. }
  14. if (data && Object.prototype.toString.call(data) !== "[object Object]") {
  15. throw new Error("data arg must be an Object");
  16. }
  17. this.userId = userId;
  18. const split = this.userId.split(":");
  19. this.localpart = split[0].substring(1);
  20. this.host = split[1];
  21. if (escape) {
  22. this.escapeUserId();
  23. }
  24. this._data = data || {};
  25. }
  26. /**
  27. * Get the matrix user's ID.
  28. * @return {string} The user ID
  29. */
  30. MatrixUser.prototype.getId = function() {
  31. return this.userId;
  32. };
  33. /**
  34. * Get the display name for this Matrix user.
  35. * @return {?string} The display name.
  36. */
  37. MatrixUser.prototype.getDisplayName = function() {
  38. return this._data.displayName;
  39. };
  40. /**
  41. * Set the display name for this Matrix user.
  42. * @param {string} name The Matrix display name.
  43. */
  44. MatrixUser.prototype.setDisplayName = function(name) {
  45. this._data.displayName = name;
  46. };
  47. /**
  48. * Set an arbitrary bridge-specific data value for this user.
  49. * @param {string} key The key to store the data value under.
  50. * @param {*} val The data value. This value should be serializable via
  51. * <code>JSON.stringify(data)</code>.
  52. */
  53. MatrixUser.prototype.set = function(key, val) {
  54. this._data[key] = val;
  55. };
  56. /**
  57. * Get the data value for the given key.
  58. * @param {string} key An arbitrary bridge-specific key.
  59. * @return {*} Stored data for this key. May be undefined.
  60. */
  61. MatrixUser.prototype.get = function(key) {
  62. return this._data[key];
  63. };
  64. /**
  65. * Serialize all the data about this user, excluding the user ID.
  66. * @return {Object} The serialised data
  67. */
  68. MatrixUser.prototype.serialize = function() {
  69. this._data.localpart = this.localpart;
  70. return this._data;
  71. };
  72. /**
  73. * Make a userId conform to the matrix spec using QP escaping.
  74. * Grammar taken from: https://matrix.org/docs/spec/appendices.html#identifier-grammar
  75. */
  76. MatrixUser.prototype.escapeUserId = function() {
  77. // NOTE: Currently Matrix accepts / in the userId, although going forward it will be removed.
  78. // NOTE: We also allow uppercase for the time being.
  79. const badChars = new Set(this.localpart.replace(/([A-Z]|[a-z]|[0-9]|-|\.|=|_)+/g, ""));
  80. let res = this.localpart;
  81. badChars.forEach((c) => {
  82. const hex = c.charCodeAt(0).toString(16).toLowerCase();
  83. res = res.replace(
  84. new RegExp(`\\${c}`, "g"),
  85. `=${hex}`
  86. );
  87. });
  88. this.localpart = res;
  89. this.userId = `@${this.localpart}:${this.host}`;
  90. };
  91. /**
  92. * @static
  93. * This is a global variable to modify the default escaping behaviour of MatrixUser.
  94. */
  95. MatrixUser.ESCAPE_DEFAULT = true;
  96. /** The MatrixUser class */
  97. module.exports = MatrixUser;