@unwanted/matrix-sdk-mini 34.13.0 → 36.0.1

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.
Files changed (96) hide show
  1. package/git-revision.txt +1 -1
  2. package/lib/@types/event.d.ts +25 -0
  3. package/lib/@types/event.d.ts.map +1 -1
  4. package/lib/@types/event.js +5 -0
  5. package/lib/@types/event.js.map +1 -1
  6. package/lib/client.d.ts +44 -19
  7. package/lib/client.d.ts.map +1 -1
  8. package/lib/client.js +148 -90
  9. package/lib/client.js.map +1 -1
  10. package/lib/embedded.d.ts +4 -3
  11. package/lib/embedded.d.ts.map +1 -1
  12. package/lib/embedded.js +57 -74
  13. package/lib/embedded.js.map +1 -1
  14. package/lib/feature.d.ts.map +1 -1
  15. package/lib/feature.js +2 -1
  16. package/lib/feature.js.map +1 -1
  17. package/lib/http-api/errors.d.ts +3 -3
  18. package/lib/http-api/errors.js +3 -3
  19. package/lib/http-api/errors.js.map +1 -1
  20. package/lib/http-api/utils.js +2 -2
  21. package/lib/http-api/utils.js.map +1 -1
  22. package/lib/models/event-timeline.d.ts.map +1 -1
  23. package/lib/models/event-timeline.js +1 -21
  24. package/lib/models/event-timeline.js.map +1 -1
  25. package/lib/models/event.d.ts +11 -1
  26. package/lib/models/event.d.ts.map +1 -1
  27. package/lib/models/event.js +48 -5
  28. package/lib/models/event.js.map +1 -1
  29. package/lib/models/invites-ignorer-types.d.ts +27 -0
  30. package/lib/models/invites-ignorer-types.d.ts.map +1 -0
  31. package/lib/models/invites-ignorer-types.js +36 -0
  32. package/lib/models/invites-ignorer-types.js.map +1 -0
  33. package/lib/models/invites-ignorer.d.ts +2 -26
  34. package/lib/models/invites-ignorer.d.ts.map +1 -1
  35. package/lib/models/invites-ignorer.js +2 -27
  36. package/lib/models/invites-ignorer.js.map +1 -1
  37. package/lib/models/room-member.d.ts +6 -1
  38. package/lib/models/room-member.d.ts.map +1 -1
  39. package/lib/models/room-member.js +7 -1
  40. package/lib/models/room-member.js.map +1 -1
  41. package/lib/models/room.d.ts +6 -1
  42. package/lib/models/room.d.ts.map +1 -1
  43. package/lib/models/room.js +7 -1
  44. package/lib/models/room.js.map +1 -1
  45. package/lib/oidc/authorize.d.ts +2 -2
  46. package/lib/oidc/authorize.d.ts.map +1 -1
  47. package/lib/oidc/authorize.js +5 -5
  48. package/lib/oidc/authorize.js.map +1 -1
  49. package/lib/oidc/discovery.d.ts +8 -0
  50. package/lib/oidc/discovery.d.ts.map +1 -1
  51. package/lib/oidc/discovery.js +22 -11
  52. package/lib/oidc/discovery.js.map +1 -1
  53. package/lib/oidc/index.d.ts +3 -4
  54. package/lib/oidc/index.d.ts.map +1 -1
  55. package/lib/oidc/index.js.map +1 -1
  56. package/lib/oidc/register.js +3 -3
  57. package/lib/oidc/register.js.map +1 -1
  58. package/lib/oidc/tokenRefresher.d.ts.map +1 -1
  59. package/lib/oidc/tokenRefresher.js +6 -5
  60. package/lib/oidc/tokenRefresher.js.map +1 -1
  61. package/lib/oidc/validate.d.ts +9 -23
  62. package/lib/oidc/validate.d.ts.map +1 -1
  63. package/lib/oidc/validate.js +13 -28
  64. package/lib/oidc/validate.js.map +1 -1
  65. package/lib/randomstring.d.ts +30 -3
  66. package/lib/randomstring.d.ts.map +1 -1
  67. package/lib/randomstring.js +68 -16
  68. package/lib/randomstring.js.map +1 -1
  69. package/lib/store/indexeddb-local-backend.js +4 -2
  70. package/lib/store/indexeddb-local-backend.js.map +1 -1
  71. package/lib/testing.d.ts +17 -0
  72. package/lib/testing.d.ts.map +1 -1
  73. package/lib/testing.js +42 -0
  74. package/lib/testing.js.map +1 -1
  75. package/package.json +5 -5
  76. package/src/@types/event.ts +17 -0
  77. package/src/client.ts +87 -41
  78. package/src/embedded.ts +41 -51
  79. package/src/feature.ts +1 -0
  80. package/src/http-api/errors.ts +3 -3
  81. package/src/http-api/utils.ts +2 -2
  82. package/src/models/event-timeline.ts +1 -21
  83. package/src/models/event.ts +46 -0
  84. package/src/models/invites-ignorer-types.ts +48 -0
  85. package/src/models/invites-ignorer.ts +7 -42
  86. package/src/models/room-member.ts +16 -1
  87. package/src/models/room.ts +16 -1
  88. package/src/oidc/authorize.ts +7 -7
  89. package/src/oidc/discovery.ts +16 -10
  90. package/src/oidc/index.ts +3 -4
  91. package/src/oidc/register.ts +3 -3
  92. package/src/oidc/tokenRefresher.ts +3 -2
  93. package/src/oidc/validate.ts +40 -63
  94. package/src/randomstring.ts +65 -19
  95. package/src/store/indexeddb-local-backend.ts +2 -2
  96. package/src/testing.ts +41 -0
@@ -1 +1 @@
1
- {"version":3,"file":"testing.js","names":["MatrixEvent","mkMatrixEvent","opts","_opts$eventId","_opts$ts","event","type","room_id","roomId","sender","content","event_id","eventId","Math","random","origin_server_ts","ts","unsigned","stateKey","undefined","state_key","mxEvent","userId","membership","name","rawDisplayName","getAvatarUrl","getMxcAvatarUrl"],"sources":["../src/testing.ts"],"sourcesContent":["/*\nCopyright 2024 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n/**\n * This file is a secondary entrypoint for the js-sdk library, exposing utilities which might be useful for writing tests.\n *\n * In general, it should not be included in runtime applications.\n *\n * @packageDocumentation\n */\n\nimport { IContent, IEvent, IUnsigned, MatrixEvent } from \"./models/event.ts\";\nimport { RoomMember } from \"./models/room-member.ts\";\nimport { EventType } from \"./@types/event.ts\";\n\n/**\n * Create a {@link MatrixEvent}.\n *\n * @param opts - Values for the event.\n */\nexport function mkMatrixEvent(opts: {\n /** Room ID of the event. */\n roomId: string;\n\n /** The sender of the event. */\n sender: string;\n\n /** The type of the event. */\n type: EventType | string;\n\n /** Optional `state_key` for the event. If unspecified, a non-state event is created. */\n stateKey?: string;\n\n /** Optional `origin_server_ts` for the event. If unspecified, the timestamp will be set to 0. */\n ts?: number;\n\n /** Optional `event_id` for the event. If provided will be used as event ID; else an ID is generated. */\n eventId?: string;\n\n /** Content of the event. */\n content: IContent;\n\n /** Optional `unsigned` data for the event. */\n unsigned?: IUnsigned;\n}): MatrixEvent {\n const event: Partial<IEvent> = {\n type: opts.type,\n room_id: opts.roomId,\n sender: opts.sender,\n content: opts.content,\n event_id: opts.eventId ?? \"$\" + Math.random() + \"-\" + Math.random(),\n origin_server_ts: opts.ts ?? 0,\n unsigned: opts.unsigned,\n };\n if (opts.stateKey !== undefined) {\n event.state_key = opts.stateKey;\n }\n\n const mxEvent = new MatrixEvent(event);\n mxEvent.sender = {\n userId: opts.sender,\n membership: \"join\",\n name: opts.sender,\n rawDisplayName: opts.sender,\n roomId: opts.sender,\n getAvatarUrl: () => {},\n getMxcAvatarUrl: () => {},\n } as unknown as RoomMember;\n return mxEvent;\n}\n"],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,SAAsCA,WAAW,QAAQ,mBAAmB;AAI5E;AACA;AACA;AACA;AACA;AACA,OAAO,SAASC,aAAaA,CAACC,IAwB7B,EAAe;EAAA,IAAAC,aAAA,EAAAC,QAAA;EACZ,IAAMC,KAAsB,GAAG;IAC3BC,IAAI,EAAEJ,IAAI,CAACI,IAAI;IACfC,OAAO,EAAEL,IAAI,CAACM,MAAM;IACpBC,MAAM,EAAEP,IAAI,CAACO,MAAM;IACnBC,OAAO,EAAER,IAAI,CAACQ,OAAO;IACrBC,QAAQ,GAAAR,aAAA,GAAED,IAAI,CAACU,OAAO,cAAAT,aAAA,cAAAA,aAAA,GAAI,GAAG,GAAGU,IAAI,CAACC,MAAM,CAAC,CAAC,GAAG,GAAG,GAAGD,IAAI,CAACC,MAAM,CAAC,CAAC;IACnEC,gBAAgB,GAAAX,QAAA,GAAEF,IAAI,CAACc,EAAE,cAAAZ,QAAA,cAAAA,QAAA,GAAI,CAAC;IAC9Ba,QAAQ,EAAEf,IAAI,CAACe;EACnB,CAAC;EACD,IAAIf,IAAI,CAACgB,QAAQ,KAAKC,SAAS,EAAE;IAC7Bd,KAAK,CAACe,SAAS,GAAGlB,IAAI,CAACgB,QAAQ;EACnC;EAEA,IAAMG,OAAO,GAAG,IAAIrB,WAAW,CAACK,KAAK,CAAC;EACtCgB,OAAO,CAACZ,MAAM,GAAG;IACba,MAAM,EAAEpB,IAAI,CAACO,MAAM;IACnBc,UAAU,EAAE,MAAM;IAClBC,IAAI,EAAEtB,IAAI,CAACO,MAAM;IACjBgB,cAAc,EAAEvB,IAAI,CAACO,MAAM;IAC3BD,MAAM,EAAEN,IAAI,CAACO,MAAM;IACnBiB,YAAY,EAAEA,CAAA,KAAM,CAAC,CAAC;IACtBC,eAAe,EAAEA,CAAA,KAAM,CAAC;EAC5B,CAA0B;EAC1B,OAAON,OAAO;AAClB","ignoreList":[]}
1
+ {"version":3,"file":"testing.js","names":["MatrixEvent","mkMatrixEvent","opts","_opts$eventId","_opts$ts","event","type","room_id","roomId","sender","content","event_id","eventId","Math","random","origin_server_ts","ts","unsigned","stateKey","undefined","state_key","mxEvent","userId","membership","name","rawDisplayName","getAvatarUrl","getMxcAvatarUrl","makeDelegatedAuthConfig","issuer","arguments","length","additionalGrantTypes","metadata","mockOpenIdConfiguration","_objectSpread","signingKeys","revocation_endpoint","token_endpoint","authorization_endpoint","registration_endpoint","device_authorization_endpoint","jwks_uri","response_types_supported","grant_types_supported","code_challenge_methods_supported"],"sources":["../src/testing.ts"],"sourcesContent":["/*\nCopyright 2024 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\n/**\n * This file is a secondary entrypoint for the js-sdk library, exposing utilities which might be useful for writing tests.\n *\n * In general, it should not be included in runtime applications.\n *\n * @packageDocumentation\n */\n\nimport { IContent, IEvent, IUnsigned, MatrixEvent } from \"./models/event.ts\";\nimport { RoomMember } from \"./models/room-member.ts\";\nimport { EventType } from \"./@types/event.ts\";\nimport { OidcClientConfig, ValidatedAuthMetadata } from \"./oidc/index.ts\";\n\n/**\n * Create a {@link MatrixEvent}.\n *\n * @param opts - Values for the event.\n */\nexport function mkMatrixEvent(opts: {\n /** Room ID of the event. */\n roomId: string;\n\n /** The sender of the event. */\n sender: string;\n\n /** The type of the event. */\n type: EventType | string;\n\n /** Optional `state_key` for the event. If unspecified, a non-state event is created. */\n stateKey?: string;\n\n /** Optional `origin_server_ts` for the event. If unspecified, the timestamp will be set to 0. */\n ts?: number;\n\n /** Optional `event_id` for the event. If provided will be used as event ID; else an ID is generated. */\n eventId?: string;\n\n /** Content of the event. */\n content: IContent;\n\n /** Optional `unsigned` data for the event. */\n unsigned?: IUnsigned;\n}): MatrixEvent {\n const event: Partial<IEvent> = {\n type: opts.type,\n room_id: opts.roomId,\n sender: opts.sender,\n content: opts.content,\n event_id: opts.eventId ?? \"$\" + Math.random() + \"-\" + Math.random(),\n origin_server_ts: opts.ts ?? 0,\n unsigned: opts.unsigned,\n };\n if (opts.stateKey !== undefined) {\n event.state_key = opts.stateKey;\n }\n\n const mxEvent = new MatrixEvent(event);\n mxEvent.sender = {\n userId: opts.sender,\n membership: \"join\",\n name: opts.sender,\n rawDisplayName: opts.sender,\n roomId: opts.sender,\n getAvatarUrl: () => {},\n getMxcAvatarUrl: () => {},\n } as unknown as RoomMember;\n return mxEvent;\n}\n\n/**\n * Makes a valid OidcClientConfig with minimum valid values\n * @param issuer used as the base for all other urls\n * @param additionalGrantTypes to add to the default grant types\n * @returns OidcClientConfig\n * @experimental\n */\nexport const makeDelegatedAuthConfig = (\n issuer = \"https://auth.org/\",\n additionalGrantTypes: string[] = [],\n): OidcClientConfig => {\n const metadata = mockOpenIdConfiguration(issuer, additionalGrantTypes);\n return {\n ...metadata,\n signingKeys: null,\n };\n};\n/**\n * Useful for mocking <issuer>/.well-known/openid-configuration\n * @param issuer used as the base for all other urls\n * @param additionalGrantTypes to add to the default grant types\n * @returns ValidatedAuthMetadata\n * @experimental\n */\nexport const mockOpenIdConfiguration = (\n issuer = \"https://auth.org/\",\n additionalGrantTypes: string[] = [],\n): ValidatedAuthMetadata => ({\n issuer,\n revocation_endpoint: issuer + \"revoke\",\n token_endpoint: issuer + \"token\",\n authorization_endpoint: issuer + \"auth\",\n registration_endpoint: issuer + \"registration\",\n device_authorization_endpoint: issuer + \"device\",\n jwks_uri: issuer + \"jwks\",\n response_types_supported: [\"code\"],\n grant_types_supported: [\"authorization_code\", \"refresh_token\", ...additionalGrantTypes],\n code_challenge_methods_supported: [\"S256\"],\n});"],"mappings":"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","ignoreList":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@unwanted/matrix-sdk-mini",
3
- "version": "34.13.0",
3
+ "version": "36.0.1",
4
4
  "description": "Matrix Client-Server mini SDK for Javascript",
5
5
  "engines": {
6
6
  "node": ">=20.0.0"
@@ -56,7 +56,7 @@
56
56
  "jwt-decode": "^4.0.0",
57
57
  "loglevel": "^1.7.1",
58
58
  "matrix-events-sdk": "0.0.1",
59
- "matrix-widget-api": "^1.10.0",
59
+ "matrix-widget-api": "^1.12.0",
60
60
  "oidc-client-ts": "^3.0.1",
61
61
  "p-retry": "4",
62
62
  "sdp-transform": "^2.15.0",
@@ -68,7 +68,7 @@
68
68
  "@action-validator/core": "^0.6.0",
69
69
  "@babel/cli": "^7.12.10",
70
70
  "@babel/core": "^7.12.10",
71
- "@babel/eslint-parser": "^7.12.10",
71
+ "@babel/eslint-parser": "^7.26.5",
72
72
  "@babel/eslint-plugin": "^7.12.10",
73
73
  "@babel/plugin-syntax-dynamic-import": "^7.8.3",
74
74
  "@babel/plugin-transform-class-properties": "^7.12.1",
@@ -87,7 +87,7 @@
87
87
  "@types/node": "18",
88
88
  "@types/sdp-transform": "^2.4.5",
89
89
  "@types/uuid": "10",
90
- "@typescript-eslint/eslint-plugin": "^8.0.0",
90
+ "@typescript-eslint/eslint-plugin": "^8.21.0",
91
91
  "@typescript-eslint/parser": "^8.0.0",
92
92
  "babel-jest": "^29.0.0",
93
93
  "babel-plugin-search-and-replace": "^1.1.1",
@@ -122,7 +122,7 @@
122
122
  "typedoc-plugin-coverage": "^3.4.0",
123
123
  "typedoc-plugin-mdn-links": "^3.0.3",
124
124
  "typedoc-plugin-missing-exports": "^3.0.0",
125
- "typescript": "^5.4.2"
125
+ "typescript": "^5.7.3"
126
126
  },
127
127
  "@casualbot/jest-sonar-reporter": {
128
128
  "outputDirectory": "coverage",
@@ -42,6 +42,9 @@ import { XOR } from "./common.ts";
42
42
  import { ReactionEventContent, RoomMessageEventContent, StickerEventContent } from "./events.ts";
43
43
 
44
44
  import { M_POLL_END, M_POLL_START, PollEndEventContent, PollStartEventContent } from "./polls.ts";
45
+ import { LocalNotificationSettings } from "./local_notifications.ts";
46
+ import { IPushRules } from "./PushRules.ts";
47
+ import { POLICIES_ACCOUNT_EVENT_TYPE } from "../models/invites-ignorer-types.ts";
45
48
 
46
49
  export enum EventType {
47
50
  // Room state events
@@ -334,3 +337,17 @@ export interface StateEvents {
334
337
  // MSC3672
335
338
  [M_BEACON_INFO.name]: MBeaconInfoEventContent;
336
339
  }
340
+
341
+ /**
342
+ * Mapped type from event type to content type for all specified global account_data events.
343
+ */
344
+ export interface AccountDataEvents {
345
+ [EventType.PushRules]: IPushRules;
346
+ [EventType.Direct]: { [userId: string]: string[] };
347
+ [EventType.IgnoredUserList]: { [userId: string]: {} };
348
+ "m.identity_server": { base_url: string | null };
349
+ [key: `${typeof LOCAL_NOTIFICATION_SETTINGS_PREFIX.name}.${string}`]: LocalNotificationSettings;
350
+ // Invites-ignorer events
351
+ [POLICIES_ACCOUNT_EVENT_TYPE.name]: { [key: string]: any };
352
+ [POLICIES_ACCOUNT_EVENT_TYPE.altName]: { [key: string]: any };
353
+ }
package/src/client.ts CHANGED
@@ -101,6 +101,7 @@ import {
101
101
  UpdateDelayedEventAction,
102
102
  } from "./@types/requests.ts";
103
103
  import {
104
+ AccountDataEvents,
104
105
  EventType,
105
106
  LOCAL_NOTIFICATION_SETTINGS_PREFIX,
106
107
  MSC3912_RELATION_BASED_REDACTIONS_PROP,
@@ -124,7 +125,7 @@ import {
124
125
  Visibility,
125
126
  } from "./@types/partials.ts";
126
127
  import { EventMapper, eventMapperFor, MapperOpts } from "./event-mapper.ts";
127
- import { randomString } from "./randomstring.ts";
128
+ import { secureRandomString } from "./randomstring.ts";
128
129
  import { DEFAULT_TREE_POWER_LEVELS_TEMPLATE, MSC3089TreeSpace } from "./models/MSC3089TreeSpace.ts";
129
130
  import { ISignatures } from "./@types/signed.ts";
130
131
  import { IStore } from "./store/index.ts";
@@ -184,6 +185,7 @@ import { RoomMessageEventContent, StickerEventContent } from "./@types/events.ts
184
185
  import { ImageInfo } from "./@types/media.ts";
185
186
  import { Capabilities, ServerCapabilities } from "./serverCapabilities.ts";
186
187
  import { sha256 } from "./digest.ts";
188
+ import { discoverAndValidateOIDCIssuerWellKnown, OidcClientConfig, validateAuthMetadataAndKeys } from "./oidc/index.ts";
187
189
 
188
190
  export type Store = IStore;
189
191
 
@@ -346,15 +348,6 @@ export interface ICreateClientOpts {
346
348
  logger?: Logger;
347
349
  }
348
350
 
349
- export interface IMatrixClientCreateOpts extends ICreateClientOpts {
350
- /**
351
- * Whether to allow sending messages to encrypted rooms when encryption
352
- * is not available internally within this SDK. This is useful if you are using an external
353
- * E2E proxy, for example. Defaults to false.
354
- */
355
- usingExternalCrypto?: boolean;
356
- }
357
-
358
351
  export enum PendingEventOrdering {
359
352
  Chronological = "chronological",
360
353
  Detached = "detached",
@@ -1009,8 +1002,6 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1009
1002
  private readonly logger: Logger;
1010
1003
 
1011
1004
  public reEmitter = new TypedReEmitter<EmittedEvents, ClientEventHandlerMap>(this);
1012
- public olmVersion: [number, number, number] | null = null; // populated after initCrypto
1013
- public usingExternalCrypto = false;
1014
1005
  private _store!: Store;
1015
1006
  public deviceId: string | null;
1016
1007
  public credentials: { userId: string | null };
@@ -1077,7 +1068,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1077
1068
 
1078
1069
  private serverCapabilitiesService: ServerCapabilities;
1079
1070
 
1080
- public constructor(opts: IMatrixClientCreateOpts) {
1071
+ public constructor(opts: ICreateClientOpts) {
1081
1072
  super();
1082
1073
 
1083
1074
  // If a custom logger is provided, use it. Otherwise, default to the global
@@ -1091,10 +1082,9 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1091
1082
  this.idBaseUrl = opts.idBaseUrl;
1092
1083
  this.identityServer = opts.identityServer;
1093
1084
 
1094
- this.usingExternalCrypto = opts.usingExternalCrypto ?? false;
1095
1085
  this.store = opts.store || new StubStore();
1096
1086
  this.deviceId = opts.deviceId || null;
1097
- this.sessionId = randomString(10);
1087
+ this.sessionId = secureRandomString(10);
1098
1088
 
1099
1089
  const userId = opts.userId || null;
1100
1090
  this.credentials = { userId };
@@ -1598,7 +1588,10 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1598
1588
  * @returns Promise which resolves: an empty object
1599
1589
  * @returns Rejects: with an error response.
1600
1590
  */
1601
- public setAccountData(eventType: EventType | string, content: IContent): Promise<{}> {
1591
+ public setAccountData<K extends keyof AccountDataEvents>(
1592
+ eventType: K,
1593
+ content: AccountDataEvents[K] | Record<string, never>,
1594
+ ): Promise<{}> {
1602
1595
  const path = utils.encodeUri("/user/$userId/account_data/$type", {
1603
1596
  $userId: this.credentials.userId!,
1604
1597
  $type: eventType,
@@ -1613,7 +1606,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1613
1606
  * @param eventType - The event type
1614
1607
  * @returns The contents of the given account data event
1615
1608
  */
1616
- public getAccountData(eventType: string): MatrixEvent | undefined {
1609
+ public getAccountData<K extends keyof AccountDataEvents>(eventType: K): MatrixEvent | undefined {
1617
1610
  return this.store.getAccountData(eventType);
1618
1611
  }
1619
1612
 
@@ -1625,7 +1618,9 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1625
1618
  * @returns Promise which resolves: The contents of the given account data event.
1626
1619
  * @returns Rejects: with an error response.
1627
1620
  */
1628
- public async getAccountDataFromServer<T extends { [k: string]: any }>(eventType: string): Promise<T | null> {
1621
+ public async getAccountDataFromServer<K extends keyof AccountDataEvents>(
1622
+ eventType: K,
1623
+ ): Promise<AccountDataEvents[K] | null> {
1629
1624
  if (this.isInitialSyncComplete()) {
1630
1625
  const event = this.store.getAccountData(eventType);
1631
1626
  if (!event) {
@@ -1633,7 +1628,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1633
1628
  }
1634
1629
  // The network version below returns just the content, so this branch
1635
1630
  // does the same to match.
1636
- return event.getContent<T>();
1631
+ return event.getContent<AccountDataEvents[K]>();
1637
1632
  }
1638
1633
  const path = utils.encodeUri("/user/$userId/account_data/$type", {
1639
1634
  $userId: this.credentials.userId!,
@@ -1649,7 +1644,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1649
1644
  }
1650
1645
  }
1651
1646
 
1652
- public async deleteAccountData(eventType: string): Promise<void> {
1647
+ public async deleteAccountData(eventType: keyof AccountDataEvents): Promise<void> {
1653
1648
  const msc3391DeleteAccountDataServerSupport = this.canSupport.get(Feature.AccountDataDeletion);
1654
1649
  // if deletion is not supported overwrite with empty content
1655
1650
  if (msc3391DeleteAccountDataServerSupport === ServerSupport.Unsupported) {
@@ -1672,7 +1667,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1672
1667
  * @returns The array of users that are ignored (empty if none)
1673
1668
  */
1674
1669
  public getIgnoredUsers(): string[] {
1675
- const event = this.getAccountData("m.ignored_user_list");
1670
+ const event = this.getAccountData(EventType.IgnoredUserList);
1676
1671
  if (!event?.getContent()["ignored_users"]) return [];
1677
1672
  return Object.keys(event.getContent()["ignored_users"]);
1678
1673
  }
@@ -1688,7 +1683,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
1688
1683
  userIds.forEach((u) => {
1689
1684
  content.ignored_users[u] = {};
1690
1685
  });
1691
- return this.setAccountData("m.ignored_user_list", content);
1686
+ return this.setAccountData(EventType.IgnoredUserList, content);
1692
1687
  }
1693
1688
 
1694
1689
  /**
@@ -2352,7 +2347,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
2352
2347
  if (this.canSupport.get(Feature.RelationBasedRedactions) === ServerSupport.Unsupported) {
2353
2348
  throw new Error(
2354
2349
  "Server does not support relation based redactions " +
2355
- `roomId ${roomId} eventId ${eventId} txnId: ${txnId} threadId ${threadId}`,
2350
+ `roomId ${roomId} eventId ${eventId} txnId: ${txnId as string} threadId ${threadId}`,
2356
2351
  );
2357
2352
  }
2358
2353
 
@@ -4630,8 +4625,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
4630
4625
  if (room) {
4631
4626
  // Copy over a known event sender if we can
4632
4627
  for (const ev of sr.context.getTimeline()) {
4633
- const sender = room.getMember(ev.getSender()!);
4634
- if (!ev.sender && sender) ev.sender = sender;
4628
+ ev.setMetadata(room.currentState, false);
4635
4629
  }
4636
4630
  }
4637
4631
  searchResults.results.push(sr);
@@ -5170,7 +5164,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
5170
5164
  * @returns A new client secret
5171
5165
  */
5172
5166
  public generateClientSecret(): string {
5173
- return randomString(32);
5167
+ return secureRandomString(32);
5174
5168
  }
5175
5169
 
5176
5170
  /**
@@ -5410,27 +5404,33 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
5410
5404
  /**
5411
5405
  * @returns Promise which resolves to a LoginResponse object
5412
5406
  * @returns Rejects: with an error response.
5407
+ *
5408
+ * @deprecated This method has unintuitive behaviour: it updates the `MatrixClient` instance with *some* of the
5409
+ * returned credentials. Instead, call {@link loginRequest} and create a new `MatrixClient` instance using the
5410
+ * results. See https://github.com/matrix-org/matrix-js-sdk/issues/4502.
5413
5411
  */
5414
5412
  public login(loginType: LoginRequest["type"], data: Omit<LoginRequest, "type">): Promise<LoginResponse> {
5415
- return this.http
5416
- .authedRequest<LoginResponse>(Method.Post, "/login", undefined, {
5417
- ...data,
5418
- type: loginType,
5419
- })
5420
- .then((response) => {
5421
- if (response.access_token && response.user_id) {
5422
- this.http.opts.accessToken = response.access_token;
5423
- this.credentials = {
5424
- userId: response.user_id,
5425
- };
5426
- }
5427
- return response;
5428
- });
5413
+ return this.loginRequest({
5414
+ ...data,
5415
+ type: loginType,
5416
+ }).then((response) => {
5417
+ if (response.access_token && response.user_id) {
5418
+ this.http.opts.accessToken = response.access_token;
5419
+ this.credentials = {
5420
+ userId: response.user_id,
5421
+ };
5422
+ }
5423
+ return response;
5424
+ });
5429
5425
  }
5430
5426
 
5431
5427
  /**
5432
5428
  * @returns Promise which resolves to a LoginResponse object
5433
5429
  * @returns Rejects: with an error response.
5430
+ *
5431
+ * @deprecated This method has unintuitive behaviour: it updates the `MatrixClient` instance with *some* of the
5432
+ * returned credentials. Instead, call {@link loginRequest} with `data.type: "m.login.password"`, and create a new
5433
+ * `MatrixClient` instance using the results. See https://github.com/matrix-org/matrix-js-sdk/issues/4502.
5434
5434
  */
5435
5435
  public loginWithPassword(user: string, password: string): Promise<LoginResponse> {
5436
5436
  return this.login("m.login.password", {
@@ -5475,6 +5475,10 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
5475
5475
  * @param token - Login token previously received from homeserver
5476
5476
  * @returns Promise which resolves to a LoginResponse object
5477
5477
  * @returns Rejects: with an error response.
5478
+ *
5479
+ * @deprecated This method has unintuitive behaviour: it updates the `MatrixClient` instance with *some* of the
5480
+ * returned credentials. Instead, call {@link loginRequest} with `data.type: "m.login.token"`, and create a new
5481
+ * `MatrixClient` instance using the results. See https://github.com/matrix-org/matrix-js-sdk/issues/4502.
5478
5482
  */
5479
5483
  public loginWithToken(token: string): Promise<LoginResponse> {
5480
5484
  return this.login("m.login.token", {
@@ -5482,6 +5486,20 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
5482
5486
  });
5483
5487
  }
5484
5488
 
5489
+ /**
5490
+ * Sends a `POST /login` request to the server.
5491
+ *
5492
+ * If successful, this will create a new device and access token for the user.
5493
+ *
5494
+ * @see {@link MatrixClient.loginFlows} which makes a `GET /login` request.
5495
+ * @see https://spec.matrix.org/v1.13/client-server-api/#post_matrixclientv3login
5496
+ *
5497
+ * @param data - Credentials and other details for the login request.
5498
+ */
5499
+ public async loginRequest(data: LoginRequest): Promise<LoginResponse> {
5500
+ return await this.http.authedRequest<LoginResponse>(Method.Post, "/login", undefined, data);
5501
+ }
5502
+
5485
5503
  /**
5486
5504
  * Logs out the current session.
5487
5505
  * Obviously, further calls that require authorisation should fail after this
@@ -6422,7 +6440,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
6422
6440
  deviceId: string,
6423
6441
  notificationSettings: LocalNotificationSettings,
6424
6442
  ): Promise<{}> {
6425
- const key = `${LOCAL_NOTIFICATION_SETTINGS_PREFIX.name}.${deviceId}`;
6443
+ const key = `${LOCAL_NOTIFICATION_SETTINGS_PREFIX.name}.${deviceId}` as const;
6426
6444
  return this.setAccountData(key, notificationSettings);
6427
6445
  }
6428
6446
 
@@ -7384,6 +7402,7 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
7384
7402
  * @returns Resolves: A promise of an object containing the OIDC issuer if configured
7385
7403
  * @returns Rejects: when the request fails (module:http-api.MatrixError)
7386
7404
  * @experimental - part of MSC2965
7405
+ * @deprecated in favour of getAuthMetadata
7387
7406
  */
7388
7407
  public async getAuthIssuer(): Promise<{
7389
7408
  issuer: string;
@@ -7392,6 +7411,33 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
7392
7411
  prefix: ClientPrefix.Unstable + "/org.matrix.msc2965",
7393
7412
  });
7394
7413
  }
7414
+
7415
+ /**
7416
+ * Discover and validate delegated auth configuration
7417
+ * - delegated auth issuer openid-configuration is reachable
7418
+ * - delegated auth issuer openid-configuration is configured correctly for us
7419
+ * Fetches /auth_metadata falling back to legacy implementation using /auth_issuer followed by
7420
+ * https://oidc-issuer.example.com/.well-known/openid-configuration and other files linked therein.
7421
+ * When successful, validated metadata is returned
7422
+ * @returns validated authentication metadata and optionally signing keys
7423
+ * @throws when delegated auth config is invalid or unreachable
7424
+ * @experimental - part of MSC2965
7425
+ */
7426
+ public async getAuthMetadata(): Promise<OidcClientConfig> {
7427
+ let authMetadata: unknown | undefined;
7428
+ try {
7429
+ authMetadata = await this.http.request<unknown>(Method.Get, "/auth_metadata", undefined, undefined, {
7430
+ prefix: ClientPrefix.Unstable + "/org.matrix.msc2965",
7431
+ });
7432
+ } catch (e) {
7433
+ if (e instanceof MatrixError && e.errcode === "M_UNRECOGNIZED") {
7434
+ const { issuer } = await this.getAuthIssuer();
7435
+ return discoverAndValidateOIDCIssuerWellKnown(issuer);
7436
+ }
7437
+ throw e;
7438
+ }
7439
+ return validateAuthMetadataAndKeys(authMetadata);
7440
+ }
7395
7441
  }
7396
7442
 
7397
7443
  function getUnstableDelayQueryOpts(delayOpts: SendDelayedEventRequestOpts): QueryDict {
package/src/embedded.ts CHANGED
@@ -28,6 +28,7 @@ import {
28
28
  WidgetApiAction,
29
29
  IWidgetApiResponse,
30
30
  IWidgetApiResponseData,
31
+ IUpdateStateToWidgetActionRequest,
31
32
  } from "matrix-widget-api";
32
33
 
33
34
  import { MatrixEvent, IEvent, IContent } from "./models/event.ts";
@@ -42,7 +43,7 @@ import { logger } from "./logger.ts";
42
43
  import {
43
44
  MatrixClient,
44
45
  ClientEvent,
45
- IMatrixClientCreateOpts,
46
+ ICreateClientOpts,
46
47
  IStartClientOpts,
47
48
  SendToDeviceContentMap,
48
49
  IOpenIDToken,
@@ -134,6 +135,7 @@ export type EventHandlerMap = { [RoomWidgetClientEvent.PendingEventsChanged]: ()
134
135
  export class RoomWidgetClient extends MatrixClient {
135
136
  private room?: Room;
136
137
  private readonly widgetApiReady: Promise<void>;
138
+ private readonly roomStateSynced: Promise<void>;
137
139
  private lifecycle?: AbortController;
138
140
  private syncState: SyncState | null = null;
139
141
 
@@ -152,9 +154,9 @@ export class RoomWidgetClient extends MatrixClient {
152
154
  */
153
155
  public constructor(
154
156
  private readonly widgetApi: WidgetApi,
155
- private readonly capabilities: ICapabilities,
157
+ /*private readonly */capabilities: ICapabilities,
156
158
  private readonly roomId: string,
157
- opts: IMatrixClientCreateOpts,
159
+ opts: ICreateClientOpts,
158
160
  sendContentLoaded: boolean,
159
161
  ) {
160
162
  super(opts);
@@ -187,6 +189,11 @@ export class RoomWidgetClient extends MatrixClient {
187
189
  };
188
190
 
189
191
  this.widgetApiReady = new Promise<void>((resolve) => this.widgetApi.once("ready", resolve));
192
+ this.roomStateSynced = capabilities.receiveState?.length
193
+ ? new Promise<void>((resolve) =>
194
+ this.widgetApi.once(`action:${WidgetApiToWidgetAction.UpdateState}`, resolve),
195
+ )
196
+ : Promise.resolve();
190
197
 
191
198
  // Request capabilities for the functionality this client needs to support
192
199
  if (
@@ -239,6 +246,7 @@ export class RoomWidgetClient extends MatrixClient {
239
246
 
240
247
  widgetApi.on(`action:${WidgetApiToWidgetAction.SendEvent}`, this.onEvent);
241
248
  widgetApi.on(`action:${WidgetApiToWidgetAction.SendToDevice}`, this.onToDevice);
249
+ widgetApi.on(`action:${WidgetApiToWidgetAction.UpdateState}`, this.onStateUpdate);
242
250
 
243
251
  // Open communication with the host
244
252
  widgetApi.start();
@@ -274,28 +282,6 @@ export class RoomWidgetClient extends MatrixClient {
274
282
 
275
283
  await this.widgetApiReady;
276
284
 
277
- // Backfill the requested events
278
- // We only get the most recent event for every type + state key combo,
279
- // so it doesn't really matter what order we inject them in
280
- await Promise.all(
281
- this.capabilities.receiveState?.map(async ({ eventType, stateKey }) => {
282
- const rawEvents = await this.widgetApi.readStateEvents(eventType, undefined, stateKey, [this.roomId]);
283
- const events = rawEvents.map((rawEvent) => new MatrixEvent(rawEvent as Partial<IEvent>));
284
-
285
- if (this.syncApi instanceof SyncApi) {
286
- // Passing undefined for `stateAfterEventList` allows will make `injectRoomEvents` run in legacy mode
287
- // -> state events in `timelineEventList` will update the state.
288
- await this.syncApi.injectRoomEvents(this.room!, undefined, events);
289
- } else {
290
- await this.syncApi!.injectRoomEvents(this.room!, events); // Sliding Sync
291
- }
292
- events.forEach((event) => {
293
- this.emit(ClientEvent.Event, event);
294
- logger.info(`Backfilled event ${event.getId()} ${event.getType()} ${event.getStateKey()}`);
295
- });
296
- }) ?? [],
297
- );
298
-
299
285
  if (opts.clientWellKnownPollPeriod !== undefined) {
300
286
  this.clientWellKnownIntervalID = setInterval(() => {
301
287
  this.fetchClientWellKnown();
@@ -303,13 +289,15 @@ export class RoomWidgetClient extends MatrixClient {
303
289
  this.fetchClientWellKnown();
304
290
  }
305
291
 
292
+ await this.roomStateSynced;
306
293
  this.setSyncState(SyncState.Syncing);
307
- logger.info("Finished backfilling events");
294
+ logger.info("Finished initial sync");
308
295
  }
309
296
 
310
297
  public stopClient(): void {
311
298
  this.widgetApi.off(`action:${WidgetApiToWidgetAction.SendEvent}`, this.onEvent);
312
299
  this.widgetApi.off(`action:${WidgetApiToWidgetAction.SendToDevice}`, this.onToDevice);
300
+ this.widgetApi.off(`action:${WidgetApiToWidgetAction.UpdateState}`, this.onStateUpdate);
313
301
 
314
302
  super.stopClient();
315
303
  this.lifecycle!.abort(); // Signal to other async tasks that the client has stopped
@@ -473,36 +461,15 @@ export class RoomWidgetClient extends MatrixClient {
473
461
  // Only inject once we have update the txId
474
462
  await this.updateTxId(event);
475
463
 
476
- // The widget API does not tell us whether a state event came from `state_after` or not so we assume legacy behaviour for now.
477
464
  if (this.syncApi instanceof SyncApi) {
478
- // The code will want to be something like:
479
- // ```
480
- // if (!params.addToTimeline && !params.addToState) {
481
- // // Passing undefined for `stateAfterEventList` makes `injectRoomEvents` run in "legacy mode"
482
- // // -> state events part of the `timelineEventList` parameter will update the state.
483
- // this.injectRoomEvents(this.room!, [], undefined, [event]);
484
- // } else {
485
- // this.injectRoomEvents(this.room!, undefined, params.addToState ? [event] : [], params.addToTimeline ? [event] : []);
486
- // }
487
- // ```
488
-
489
- // Passing undefined for `stateAfterEventList` allows will make `injectRoomEvents` run in legacy mode
490
- // -> state events in `timelineEventList` will update the state.
491
- await this.syncApi.injectRoomEvents(this.room!, [], undefined, [event]);
465
+ await this.syncApi.injectRoomEvents(this.room!, undefined, [], [event]);
492
466
  } else {
493
- // The code will want to be something like:
494
- // ```
495
- // if (!params.addToTimeline && !params.addToState) {
496
- // this.injectRoomEvents(this.room!, [], [event]);
497
- // } else {
498
- // this.injectRoomEvents(this.room!, params.addToState ? [event] : [], params.addToTimeline ? [event] : []);
499
- // }
500
- // ```
501
- await this.syncApi!.injectRoomEvents(this.room!, [], [event]); // Sliding Sync
467
+ // Sliding Sync
468
+ await this.syncApi!.injectRoomEvents(this.room!, [], [event]);
502
469
  }
503
470
  this.emit(ClientEvent.Event, event);
504
471
  this.setSyncState(SyncState.Syncing);
505
- logger.info(`Received event ${event.getId()} ${event.getType()} ${event.getStateKey()}`);
472
+ logger.info(`Received event ${event.getId()} ${event.getType()}`);
506
473
  } else {
507
474
  const { event_id: eventId, room_id: roomId } = ev.detail.data;
508
475
  logger.info(`Received event ${eventId} for a different room ${roomId}; discarding`);
@@ -524,6 +491,29 @@ export class RoomWidgetClient extends MatrixClient {
524
491
  this.setSyncState(SyncState.Syncing);
525
492
  await this.ack(ev);
526
493
  };
494
+
495
+ private onStateUpdate = async (ev: CustomEvent<IUpdateStateToWidgetActionRequest>): Promise<void> => {
496
+ ev.preventDefault();
497
+ for (const rawEvent of ev.detail.data.state) {
498
+ // Verify the room ID matches, since it's possible for the client to
499
+ // send us state updates from other rooms if this widget is always
500
+ // on screen
501
+ if (rawEvent.room_id === this.roomId) {
502
+ const event = new MatrixEvent(rawEvent as Partial<IEvent>);
503
+ if (this.syncApi instanceof SyncApi) {
504
+ await this.syncApi.injectRoomEvents(this.room!, undefined, [event]);
505
+ } else {
506
+ // Sliding Sync
507
+ await this.syncApi!.injectRoomEvents(this.room!, [event]);
508
+ }
509
+ logger.info(`Updated state entry ${event.getType()} ${event.getStateKey()} to ${event.getId()}`);
510
+ } else {
511
+ const { event_id: eventId, room_id: roomId } = ev.detail.data;
512
+ logger.info(`Received state entry ${eventId} for a different room ${roomId}; discarding`);
513
+ }
514
+ }
515
+ await this.ack(ev);
516
+ };
527
517
  }
528
518
 
529
519
  function processAndThrow(error: unknown): never {
package/src/feature.ts CHANGED
@@ -60,6 +60,7 @@ const featureSupportResolver: Record<string, FeatureSupportCondition> = {
60
60
  },
61
61
  [Feature.RelationsRecursion]: {
62
62
  unstablePrefixes: ["org.matrix.msc3981"],
63
+ matrixVersion: "v1.10",
63
64
  },
64
65
  [Feature.IntentionalMentions]: {
65
66
  unstablePrefixes: ["org.matrix.msc3952_intentional_mentions"],
@@ -167,9 +167,9 @@ export class MatrixError extends HTTPError {
167
167
  }
168
168
 
169
169
  /**
170
- * @returns The recommended delay in milliseconds to wait before retrying
171
- * the request that triggered {@link error}, or {@link defaultMs} if the
172
- * error was not due to rate-limiting or if no valid delay is recommended.
170
+ * @returns The recommended delay in milliseconds to wait before retrying the request.
171
+ * @param error - The error to check for a retry delay.
172
+ * @param defaultMs - The delay to use if the error was not due to rate-limiting or if no valid delay is recommended.
173
173
  */
174
174
  export function safeGetRetryAfterMs(error: unknown, defaultMs: number): number {
175
175
  if (!(error instanceof HTTPError) || !error.isRateLimitError()) {
@@ -180,8 +180,8 @@ export function calculateRetryBackoff(err: any, attempts: number, retryConnectio
180
180
  return -1;
181
181
  }
182
182
 
183
- if (err.httpStatus && (err.httpStatus === 400 || err.httpStatus === 403 || err.httpStatus === 401)) {
184
- // client error; no amount of retrying will save you now.
183
+ if (err.httpStatus && Math.floor(err.httpStatus / 100) === 4 && err.httpStatus !== 429) {
184
+ // client error; no amount of retrying will save you now (except for rate limiting which is handled below)
185
185
  return -1;
186
186
  }
187
187
 
@@ -68,27 +68,7 @@ export class EventTimeline {
68
68
  * @param toStartOfTimeline - if true the event's forwardLooking flag is set false
69
69
  */
70
70
  public static setEventMetadata(event: MatrixEvent, stateContext: RoomState, toStartOfTimeline: boolean): void {
71
- // When we try to generate a sentinel member before we have that member
72
- // in the members object, we still generate a sentinel but it doesn't
73
- // have a membership event, so test to see if events.member is set. We
74
- // check this to avoid overriding non-sentinel members by sentinel ones
75
- // when adding the event to a filtered timeline
76
- if (!event.sender?.events?.member) {
77
- event.sender = stateContext.getSentinelMember(event.getSender()!);
78
- }
79
- if (!event.target?.events?.member && event.getType() === EventType.RoomMember) {
80
- event.target = stateContext.getSentinelMember(event.getStateKey()!);
81
- }
82
-
83
- if (event.isState()) {
84
- // room state has no concept of 'old' or 'current', but we want the
85
- // room state to regress back to previous values if toStartOfTimeline
86
- // is set, which means inspecting prev_content if it exists. This
87
- // is done by toggling the forwardLooking flag.
88
- if (toStartOfTimeline) {
89
- event.forwardLooking = false;
90
- }
91
- }
71
+ event.setMetadata(stateContext, toStartOfTimeline);
92
72
  }
93
73
 
94
74
  private readonly roomId: string | null;