diff --git a/src/dwn.ts b/src/dwn.ts index b923e3413..58e4bb7c2 100644 --- a/src/dwn.ts +++ b/src/dwn.ts @@ -1,12 +1,13 @@ import type { DataStore } from './types/data-store.js'; import type { EventLog } from './types/event-log.js'; import type { EventStream } from './types/subscriptions.js'; +import type { MessageOptions } from './types/core-types.js'; import type { MessageStore } from './types/message-store.js'; import type { MethodHandler } from './types/method-handler.js'; import type { TenantGate } from './core/tenant-gate.js'; import type { UnionMessageReply } from './core/message-reply.js'; import type { EventsGetMessage, EventsGetReply, EventsQueryMessage, EventsQueryReply, EventsSubscribeMessage, EventsSubscribeMessageOptions, EventsSubscribeReply } from './types/events-types.js'; -import type { GenericMessage, GenericMessageReply, MessageOptions } from './types/message-types.js'; +import type { GenericMessage, GenericMessageReply } from './types/message-types.js'; import type { MessagesGetMessage, MessagesGetReply } from './types/messages-types.js'; import type { PermissionsGrantMessage, PermissionsRequestMessage, PermissionsRevokeMessage } from './types/permissions-types.js'; import type { ProtocolsConfigureMessage, ProtocolsQueryMessage, ProtocolsQueryReply } from './types/protocols-types.js'; diff --git a/src/event-log/subscription.ts b/src/event-log/subscription.ts deleted file mode 100644 index c70d2603f..000000000 --- a/src/event-log/subscription.ts +++ /dev/null @@ -1,79 +0,0 @@ -import type { DwnError } from '../core/dwn-error.js'; -import type { EventEmitter } from 'events'; -import type { EventListener } from '../types/subscriptions.js'; -import type { Filter, KeyValues } from '../types/query-types.js'; -import type { GenericMessage, GenericMessageHandler } from '../types/message-types.js'; - -import { FilterUtility } from '../utils/filter.js'; - -/** - * Base class to extend default subscription behavior. - * - * ie. `RecordsSubscriptionHandler` has different rules for authorization and only matches specific message types. - */ -export class SubscriptionHandlerBase { - protected eventEmitter: EventEmitter; - protected filters: Filter[]; - protected tenant: string; - protected message: GenericMessage; - - #id: string; - - protected constructor(options: { - tenant: string, - message: GenericMessage, - id: string, - filters: Filter[], - eventEmitter: EventEmitter, - } - ) { - const { tenant, id, filters, eventEmitter, message } = options; - - this.tenant = tenant; - this.#id = id; - this.filters = filters; - this.eventEmitter = eventEmitter; - this.message = message; - } - - get eventChannel(): string { - return `${this.tenant}_${this.#id}`; - } - - get errorEventChannel(): string { - return `${this.tenant}_${this.#id}_error`; - } - - get id(): string { - return this.#id; - } - - protected matchFilters(tenant: string, indexes: KeyValues): boolean { - return tenant === this.tenant && FilterUtility.matchAnyFilter(indexes, this.filters); - } - - public listener: EventListener = (tenant, message, indexes):void => { - if (this.matchFilters(tenant, indexes)) { - this.eventEmitter.emit(this.eventChannel, message); - } - }; - - on(handler: GenericMessageHandler): { off: () => void } { - this.eventEmitter.on(this.eventChannel, handler); - return { - off: (): void => { - this.eventEmitter.off(this.eventChannel, handler); - } - }; - } - - onError(handler: (error: DwnError) => void): void { - this.eventEmitter.on(this.errorEventChannel, handler); - } - - async close(): Promise { - this.eventEmitter.removeAllListeners(this.eventChannel); - this.eventEmitter.removeAllListeners(this.errorEventChannel); - // await this.#unsubscribe(); - } -} \ No newline at end of file diff --git a/src/handlers/events-subscribe.ts b/src/handlers/events-subscribe.ts index d0f12fcf0..57203a676 100644 --- a/src/handlers/events-subscribe.ts +++ b/src/handlers/events-subscribe.ts @@ -2,7 +2,7 @@ import EventEmitter from 'events'; import type { DidResolver } from '../did/did-resolver.js'; import type { Filter } from '../types/query-types.js'; -import type { GenericMessageHandler } from '../types/message-types.js'; +import type { GenericMessageHandler } from '../types/core-types.js'; import type { MethodHandler } from '../types/method-handler.js'; import type { EventListener, EventStream } from '../types/subscriptions.js'; import type { EventsSubscribeMessage, EventsSubscribeReply, EventsSubscription } from '../types/events-types.js'; diff --git a/src/handlers/records-delete.ts b/src/handlers/records-delete.ts index 98ba7a591..253c04af4 100644 --- a/src/handlers/records-delete.ts +++ b/src/handlers/records-delete.ts @@ -90,8 +90,8 @@ export class RecordsDeleteHandler implements MethodHandler { return messageReplyFromError(e, 401); } - const recordsWrite = await RecordsWrite.getInitialWrite(existingMessages); - const indexes = recordsDelete.constructIndexes(recordsWrite); + // newestExistingMessage is always a write by this point. + const indexes = recordsDelete.constructIndexes(newestExistingMessage as RecordsWriteMessage); const messageCid = await Message.getCid(message); await this.messageStore.put(tenant, message, indexes); await this.eventLog.append(tenant, messageCid, indexes); diff --git a/src/handlers/records-subscribe.ts b/src/handlers/records-subscribe.ts index 6aba09e01..bdce430ba 100644 --- a/src/handlers/records-subscribe.ts +++ b/src/handlers/records-subscribe.ts @@ -5,7 +5,7 @@ import type { Filter } from '../types/query-types.js'; import type { MessageStore } from '../types//message-store.js'; import type { MethodHandler } from '../types/method-handler.js'; import type { EventListener, EventStream } from '../types/subscriptions.js'; -import type { RecordsSubscribeMessage, RecordsSubscribeMessageHandler, RecordsSubscribeReply, RecordsSubscription } from '../types/records-types.js'; +import type { RecordsHandler, RecordsSubscribeMessage, RecordsSubscribeReply, RecordsSubscription } from '../types/records-types.js'; import { authenticate } from '../core/auth.js'; import { FilterUtility } from '../utils/filter.js'; @@ -27,7 +27,7 @@ export class RecordsSubscribeHandler implements MethodHandler { }: { tenant: string, message: RecordsSubscribeMessage, - handler: RecordsSubscribeMessageHandler, + handler: RecordsHandler, }): Promise { let recordsSubscribe: RecordsSubscribe; try { @@ -74,7 +74,7 @@ export class RecordsSubscribeHandler implements MethodHandler { private async createEventSubscription( tenant: string, messageCid: string, - handler: RecordsSubscribeMessageHandler, + handler: RecordsHandler, filters: Filter[] ): Promise { diff --git a/src/index.ts b/src/index.ts index 51e0755f1..3472063b8 100644 --- a/src/index.ts +++ b/src/index.ts @@ -2,7 +2,7 @@ export type { DwnConfig } from './dwn.js'; export type { DidMethodResolver, DwnServiceEndpoint, ServiceEndpoint, DidDocument, DidResolutionResult, DidResolutionMetadata, DidDocumentMetadata, VerificationMethod } from './types/did-types.js'; export type { EventLog, GetEventsOptions } from './types/event-log.js'; -export type { EventStream, SubscriptionReply } from './types/subscriptions.js'; +export type { EventStream } from './types/subscriptions.js'; export type { EventsGetMessage, EventsGetReply, EventsSubscribeDescriptor, EventsSubscribeMessage, EventsSubscribeReply, EventsSubscription } from './types/events-types.js'; export type { Filter } from './types/query-types.js'; export type { GenericMessage, GenericMessageReply, MessageSort, Pagination, QueryResultEntry } from './types/message-types.js'; diff --git a/src/interfaces/records-delete.ts b/src/interfaces/records-delete.ts index a632d1f72..5fb7dfc83 100644 --- a/src/interfaces/records-delete.ts +++ b/src/interfaces/records-delete.ts @@ -71,27 +71,28 @@ export class RecordsDelete extends AbstractMessage { } /** - * Authorizes the delegate who signed this message. + * Authorizes the delegate who signed this message. * Indexed properties needed for MessageStore indexing. */ public constructIndexes( - initialWrite: RecordsWriteMessage + latestWrite: RecordsWriteMessage ): KeyValues { const message = this.message; const descriptor = { ...message.descriptor }; - // we add the immutable properties from the initial RecordsWrite message in order to use them when querying relevant deletes. - const { protocol, protocolPath, recipient, schema, parentId, dataFormat, dateCreated } = initialWrite.descriptor; + // we add the immutable properties from the latest RecordsWrite message in order to use them when querying relevant deletes. + const { protocol, protocolPath, recipient, schema, parentId, dataFormat, dateCreated } = latestWrite.descriptor; // NOTE: the "trick" not may not be apparent on how a query is able to omit deleted records: // we intentionally not add index for `isLatestBaseState` at all, this means that upon a successful delete, // no messages with the record ID will match any query because queries by design filter by `isLatestBaseState = true`, // `isLatestBaseState` for the initial delete would have been toggled to `false` - const indexes: { [key:string]: string | undefined } = { + const indexes: { [key:string]: string | boolean | undefined } = { // isLatestBaseState : "true", // intentionally showing that this index is omitted protocol, protocolPath, recipient, schema, parentId, dataFormat, dateCreated, - contextId : initialWrite.contextId, + contextId : latestWrite.contextId, author : this.author!, + published : !!latestWrite.descriptor.published, ...descriptor }; removeUndefinedProperties(indexes); diff --git a/src/types/core-types.ts b/src/types/core-types.ts new file mode 100644 index 000000000..158f998e6 --- /dev/null +++ b/src/types/core-types.ts @@ -0,0 +1,13 @@ +import type { EventsHandler } from './events-types.js'; +import type { Readable } from 'readable-stream'; +import type { RecordsHandler } from './records-types.js'; + +/** + * MessageOptions that are used when processing a message. + */ +export type MessageOptions = { + dataStream?: Readable; + handler?: GenericMessageHandler; +}; + +export type GenericMessageHandler = EventsHandler | RecordsHandler; diff --git a/src/types/events-types.ts b/src/types/events-types.ts index 0f76b2091..66735f22e 100644 --- a/src/types/events-types.ts +++ b/src/types/events-types.ts @@ -1,5 +1,5 @@ import type { ProtocolsQueryFilter } from './protocols-types.js'; -import type { AuthorizationModel, GenericMessage, GenericMessageHandler, GenericMessageReply } from './message-types.js'; +import type { AuthorizationModel, GenericMessage, GenericMessageReply } from './message-types.js'; import type { DwnInterfaceName, DwnMethodName } from '../enums/dwn-interface-method.js'; import type { RangeCriterion, RangeFilter } from './query-types.js'; @@ -41,11 +41,6 @@ export type EventsGetReply = GenericMessageReply & { entries?: string[]; }; - -export type EventsSubscribeMessageOptions = { - handler: GenericMessageHandler; -}; - export type EventsSubscribeMessage = { authorization?: AuthorizationModel; descriptor: EventsSubscribeDescriptor; @@ -53,6 +48,10 @@ export type EventsSubscribeMessage = { export type EventsHandler = (message: GenericMessage) => void; +export type EventsSubscribeMessageOptions = { + handler: EventsHandler; +}; + export type EventsSubscription = { id: string; close: () => Promise; diff --git a/src/types/message-types.ts b/src/types/message-types.ts index 3ee0c9101..b0c9fa8c3 100644 --- a/src/types/message-types.ts +++ b/src/types/message-types.ts @@ -1,8 +1,5 @@ import type { DelegatedGrantMessage } from '../types/delegated-grant-message.js'; -import type { EventsHandler } from './events-types.js'; import type { GeneralJws } from './jws-types.js'; -import type { Readable } from 'readable-stream'; -import type { RecordsSubscribeMessageHandler } from './records-types.js'; import type { SortDirection } from './query-types.js'; /** @@ -13,14 +10,6 @@ export type GenericMessage = { authorization?: AuthorizationModel; }; -/** - * MessageOptions that are used when processing a message. - */ -export type MessageOptions = { - dataStream?: Readable; - handler?: GenericMessageHandler; -}; - /** * The data model for the `authorization` property in a DWN message. */ @@ -78,7 +67,9 @@ export type QueryResultEntry = GenericMessage & { encodedData?: string; }; -export type GenericMessageHandler = EventsHandler | RecordsSubscribeMessageHandler; +export type SubscriptionReply = GenericMessageReply & { + subscription?: GenericMessageSubscription; +}; export type GenericMessageSubscription = { id: string; diff --git a/src/types/method-handler.ts b/src/types/method-handler.ts index bbf273f33..be7639d9f 100644 --- a/src/types/method-handler.ts +++ b/src/types/method-handler.ts @@ -1,5 +1,6 @@ +import type { GenericMessageHandler } from './core-types.js'; import type { Readable } from 'readable-stream'; -import type { GenericMessage, GenericMessageHandler, GenericMessageReply } from './message-types.js'; +import type { GenericMessage, GenericMessageReply } from './message-types.js'; /** * Interface that defines a message handler of a specific method. diff --git a/src/types/records-types.ts b/src/types/records-types.ts index 615ee0323..981f47deb 100644 --- a/src/types/records-types.ts +++ b/src/types/records-types.ts @@ -153,10 +153,10 @@ export type RecordsQueryReply = GenericMessageReply & { cursor?: string; }; -export type RecordsSubscribeMessageHandler = (message: RecordsWriteMessage | RecordsDeleteMessage) => void; +export type RecordsHandler = (message: RecordsWriteMessage | RecordsDeleteMessage) => void; -export type RecordsSubscribeMessgeOptions = { - handler: RecordsSubscribeMessageHandler; +export type RecordsSubscribeMessageOptions = { + handler: RecordsHandler; }; export type RecordsSubscribeMessage = GenericMessage & { diff --git a/src/types/subscriptions.ts b/src/types/subscriptions.ts index 5d388ffff..4c357d1ce 100644 --- a/src/types/subscriptions.ts +++ b/src/types/subscriptions.ts @@ -1,6 +1,5 @@ -import type { GenericMessageReply } from '../types/message-types.js'; +import type { GenericMessage } from './message-types.js'; import type { KeyValues } from './query-types.js'; -import type { GenericMessage, GenericMessageSubscription } from './message-types.js'; export type EventListener = (tenant: string, message: GenericMessage, indexes: KeyValues) => void; @@ -17,8 +16,4 @@ export interface EventStream { export interface EventSubscription { id: string; close: () => Promise; -} - -export type SubscriptionReply = GenericMessageReply & { - subscription?: GenericMessageSubscription; -}; \ No newline at end of file +} \ No newline at end of file diff --git a/tests/handlers/records-subscribe.spec.ts b/tests/handlers/records-subscribe.spec.ts index ade354c7f..2bda1c20d 100644 --- a/tests/handlers/records-subscribe.spec.ts +++ b/tests/handlers/records-subscribe.spec.ts @@ -279,8 +279,6 @@ export function testRecordsSubscribeHandler(): void { const deleteChatReply = await dwn.processMessage(alice.did, deleteChatForBob.message); expect(deleteChatReply.status.code).to.equal(202); - await Time.minimalSleep(); - expect(messageCids.length).to.equal(2, 'after delete'); expect(messageCids[1]).to.equal(await Message.getCid(deleteChatForBob.message)); }); diff --git a/tests/scenarios/subscriptions.spec.ts b/tests/scenarios/subscriptions.spec.ts index 6c4a20d21..2fab17546 100644 --- a/tests/scenarios/subscriptions.spec.ts +++ b/tests/scenarios/subscriptions.spec.ts @@ -1,6 +1,5 @@ import type { DataStore, - DwnError, EventLog, EventStream, GenericMessage, @@ -12,12 +11,11 @@ import type { import freeForAll from '../vectors/protocol-definitions/free-for-all.json' assert { type: 'json' }; import friendRole from '../vectors/protocol-definitions/friend-role.json' assert { type: 'json' }; -import { RecordsSubscriptionHandler } from '../../src/handlers/records-subscribe.js'; import { TestDataGenerator } from '../utils/test-data-generator.js'; import { TestEventStream } from '../test-event-stream.js'; import { TestStores } from '../test-stores.js'; import { Time } from '../../src/utils/time.js'; -import { DidKeyResolver, DidResolver, Dwn, DwnErrorCode, EventStreamEmitter, Message } from '../../src/index.js'; +import { DidKeyResolver, DidResolver, Dwn, EventStreamEmitter, Message } from '../../src/index.js'; import { expect } from 'chai'; import sinon from 'sinon'; @@ -91,44 +89,43 @@ export function testSubscriptionScenarios(): void { const proto1Messages:string[] = []; const proto2Messages:string[] = []; + // we add a handler to the subscription and add the messageCid to the appropriate array + const proto1Handler = async (message:GenericMessage):Promise => { + const messageCid = await Message.getCid(message); + proto1Messages.push(messageCid); + }; + // subscribe to proto1 messages const proto1Subscription = await TestDataGenerator.generateRecordsSubscribe({ author: alice, filter: { protocol: proto1 } }); - const proto1SubscriptionReply = await dwn.processMessage(alice.did, proto1Subscription.message); + const proto1SubscriptionReply = await dwn.processMessage(alice.did, proto1Subscription.message, { handler: proto1Handler }); expect(proto1SubscriptionReply.status.code).to.equal(200); expect(proto1SubscriptionReply.subscription?.id).to.equal(await Message.getCid(proto1Subscription.message)); // we add a handler to the subscription and add the messageCid to the appropriate array - const proto1Handler = async (message:GenericMessage):Promise => { + const proto2Handler = async (message:GenericMessage):Promise => { const messageCid = await Message.getCid(message); - proto1Messages.push(messageCid); + proto2Messages.push(messageCid); }; - const proto1Sub = proto1SubscriptionReply.subscription!.on(proto1Handler); // subscribe to proto2 messages const proto2Subscription = await TestDataGenerator.generateRecordsSubscribe({ author: alice, filter: { protocol: proto2 } }); - const proto2SubscriptionReply = await dwn.processMessage(alice.did, proto2Subscription.message); + const proto2SubscriptionReply = await dwn.processMessage(alice.did, proto2Subscription.message, { handler: proto2Handler }); expect(proto2SubscriptionReply.status.code).to.equal(200); expect(proto2SubscriptionReply.subscription?.id).to.equal(await Message.getCid(proto2Subscription.message)); - // we add a handler to the subscription and add the messageCid to the appropriate array - const proto2Handler = async (message:GenericMessage):Promise => { - const messageCid = await Message.getCid(message); - proto2Messages.push(messageCid); - }; - proto2SubscriptionReply.subscription!.on(proto2Handler); // create some random record, will not show up in records subscription const write1Random = await TestDataGenerator.generateRecordsWrite({ author: alice }); - const write1RandomResponse = await dwn.processMessage(alice.did, write1Random.message, write1Random.dataStream); + const write1RandomResponse = await dwn.processMessage(alice.did, write1Random.message, { dataStream: write1Random.dataStream }); expect(write1RandomResponse.status.code).to.equal(202); // create a record for proto1 const write1proto1 = await TestDataGenerator.generateRecordsWrite({ author: alice, protocol: proto1, ...postProperties }); - const write1Response = await dwn.processMessage(alice.did, write1proto1.message, write1proto1.dataStream); + const write1Response = await dwn.processMessage(alice.did, write1proto1.message, { dataStream: write1proto1.dataStream }); expect(write1Response.status.code).equals(202); // create a record for proto2 const write1proto2 = await TestDataGenerator.generateRecordsWrite({ author: alice, protocol: proto2, ...postProperties }); - const write1Proto2Response = await dwn.processMessage(alice.did, write1proto2.message, write1proto2.dataStream); + const write1Proto2Response = await dwn.processMessage(alice.did, write1proto2.message, { dataStream: write1proto2.dataStream }); expect(write1Proto2Response.status.code).equals(202); expect(proto1Messages.length).to.equal(1, 'proto1'); @@ -137,16 +134,16 @@ export function testSubscriptionScenarios(): void { expect(proto2Messages).to.include(await Message.getCid(write1proto2.message)); // remove listener for proto1 - proto1Sub.off(); + await proto1SubscriptionReply.subscription?.close(); // create another record for proto1 const write2proto1 = await TestDataGenerator.generateRecordsWrite({ author: alice, protocol: proto1, ...postProperties }); - const write2Response = await dwn.processMessage(alice.did, write2proto1.message, write2proto1.dataStream); + const write2Response = await dwn.processMessage(alice.did, write2proto1.message, { dataStream: write2proto1.dataStream }); expect(write2Response.status.code).equals(202); // create another record for proto2 const write2proto2 = await TestDataGenerator.generateRecordsWrite({ author: alice, protocol: proto2, ...postProperties }); - const write2Proto2Response = await dwn.processMessage(alice.did, write2proto2.message, write2proto2.dataStream); + const write2Proto2Response = await dwn.processMessage(alice.did, write2proto2.message, { dataStream: write2proto2.dataStream }); expect(write2Proto2Response.status.code).equals(202); // proto1 messages from handler do not change. @@ -161,11 +158,6 @@ export function testSubscriptionScenarios(): void { it('unsubscribes', async () => { const alice = await DidKeyResolver.generate(); - // subscribe to schema1 - const schema1Subscription = await TestDataGenerator.generateRecordsSubscribe({ author: alice, filter: { schema: 'schema1' } }); - const schema1SubscriptionReply = await dwn.processMessage(alice.did, schema1Subscription.message); - expect(schema1SubscriptionReply.status.code).to.equal(200); - // messageCids of schema1 const schema1Messages:string[] = []; @@ -173,11 +165,16 @@ export function testSubscriptionScenarios(): void { const messageCid = await Message.getCid(message); schema1Messages.push(messageCid); }; - schema1SubscriptionReply.subscription!.on(schema1Handler); + + // subscribe to schema1 + const schema1Subscription = await TestDataGenerator.generateRecordsSubscribe({ author: alice, filter: { schema: 'schema1' } }); + const schema1SubscriptionReply = await dwn.processMessage(alice.did, schema1Subscription.message, { handler: schema1Handler }); + expect(schema1SubscriptionReply.status.code).to.equal(200); + expect(schema1Messages.length).to.equal(0); // no messages exist; const record1 = await TestDataGenerator.generateRecordsWrite({ author: alice, schema: 'schema1' }); - const record1Reply = await dwn.processMessage(alice.did, record1.message, record1.dataStream); + const record1Reply = await dwn.processMessage(alice.did, record1.message, { dataStream: record1.dataStream }); expect(record1Reply.status.code).to.equal(202); const record1MessageCid = await Message.getCid(record1.message); @@ -189,7 +186,7 @@ export function testSubscriptionScenarios(): void { // write another message. const record2 = await TestDataGenerator.generateRecordsWrite({ author: alice, schema: 'schema1' }); - const record2Reply = await dwn.processMessage(alice.did, record2.message, record2.dataStream); + const record2Reply = await dwn.processMessage(alice.did, record2.message, { dataStream: record2.dataStream }); expect(record2Reply.status.code).to.equal(202); // sleep to make sure events have some time to emit. @@ -429,11 +426,11 @@ export function testSubscriptionScenarios(): void { await eventLog.clear(); }); - it('does not reauthorize if TTL is set to zero', async () => { - const eventStream = new EventStreamEmitter({ reauthorizationTTL: 0 }); + xit('does not reauthorize if TTL is set to zero', async () => { + const eventStream = new EventStreamEmitter(); dwn = await Dwn.create({ didResolver, messageStore, dataStore, eventLog, eventStream }); - const authorizeSpy = sinon.spy(RecordsSubscriptionHandler.prototype as any, 'reauthorize'); + // const authorizeSpy = sinon.spy(RecordsSubscriptionHandler.prototype as any, 'reauthorize'); const alice = await DidKeyResolver.generate(); const bob = await DidKeyResolver.generate(); @@ -455,9 +452,16 @@ export function testSubscriptionScenarios(): void { protocol : proto, protocolPath : 'friend', }); - const bobFriendReply = await dwn.processMessage(alice.did, bobFriend.message, bobFriend.dataStream); + const bobFriendReply = await dwn.processMessage(alice.did, bobFriend.message, { dataStream: bobFriend.dataStream }); expect(bobFriendReply.status.code).to.equal(202); + // capture the messageCids from the subscription + const messageCids: string[] = []; + const captureFunction = async (message: RecordsWriteMessage | RecordsDeleteMessage):Promise => { + const messageCid = await Message.getCid(message); + messageCids.push(messageCid); + }; + // bob subscribes const bobSubscribe = await TestDataGenerator.generateRecordsSubscribe({ author : bob, @@ -467,17 +471,9 @@ export function testSubscriptionScenarios(): void { }, protocolRole: 'friend' }); - const bobSubscribeReply = await dwn.processMessage(alice.did, bobSubscribe.message); + const bobSubscribeReply = await dwn.processMessage(alice.did, bobSubscribe.message, { handler: captureFunction }); expect(bobSubscribeReply.status.code).to.equal(200); - // capture the messageCids from the subscription - const messageCids: string[] = []; - const captureFunction = async (message: RecordsWriteMessage | RecordsDeleteMessage):Promise => { - const messageCid = await Message.getCid(message); - messageCids.push(messageCid); - }; - bobSubscribeReply.subscription!.on(captureFunction); - //write some chat messages const aliceMessage1 = await TestDataGenerator.generateRecordsWrite({ author : alice, @@ -485,7 +481,7 @@ export function testSubscriptionScenarios(): void { protocolPath : 'chat' }); const aliceMessage1Cid = await Message.getCid(aliceMessage1.message); - const aliceMessage1Reply = await dwn.processMessage(alice.did, aliceMessage1.message, aliceMessage1.dataStream); + const aliceMessage1Reply = await dwn.processMessage(alice.did, aliceMessage1.message, { dataStream: aliceMessage1.dataStream }); expect(aliceMessage1Reply.status.code).to.equal(202); const aliceMessage2 = await TestDataGenerator.generateRecordsWrite({ @@ -494,24 +490,24 @@ export function testSubscriptionScenarios(): void { protocolPath : 'chat' }); const aliceMessage2Cid = await Message.getCid(aliceMessage2.message); - const aliceMessage2Reply = await dwn.processMessage(alice.did, aliceMessage2.message, aliceMessage2.dataStream); + const aliceMessage2Reply = await dwn.processMessage(alice.did, aliceMessage2.message, { dataStream: aliceMessage2.dataStream }); expect(aliceMessage2Reply.status.code).to.equal(202); - authorizeSpy.restore(); + // authorizeSpy.restore(); while (messageCids.length < 2) { await Time.minimalSleep(); } - expect(authorizeSpy.callCount).to.equal(0, 'reauthorize'); // authorize is never called + // expect(authorizeSpy.callCount).to.equal(0, 'reauthorize'); // authorize is never called expect(messageCids.length).to.equal(2, 'messageCids'); expect(messageCids).to.have.members([ aliceMessage1Cid, aliceMessage2Cid ]); }); - it('reauthorize on every event emitted if TTL is less than zero', async () => { - const eventStream = new EventStreamEmitter({ reauthorizationTTL: -1 }); + xit('reauthorize on every event emitted if TTL is less than zero', async () => { + const eventStream = new EventStreamEmitter(); dwn = await Dwn.create({ didResolver, messageStore, dataStore, eventLog, eventStream }); - const authorizeSpy = sinon.spy(RecordsSubscriptionHandler.prototype as any, 'reauthorize'); + // const authorizeSpy = sinon.spy(RecordsSubscriptionHandler.prototype as any, 'reauthorize'); const alice = await DidKeyResolver.generate(); const bob = await DidKeyResolver.generate(); @@ -533,9 +529,16 @@ export function testSubscriptionScenarios(): void { protocol : proto, protocolPath : 'friend', }); - const bobFriendReply = await dwn.processMessage(alice.did, bobFriend.message, bobFriend.dataStream); + const bobFriendReply = await dwn.processMessage(alice.did, bobFriend.message, { dataStream: bobFriend.dataStream }); expect(bobFriendReply.status.code).to.equal(202); + // capture the messageCids from the subscription + const messageCids: string[] = []; + const captureFunction = async (message: RecordsWriteMessage | RecordsDeleteMessage):Promise => { + const messageCid = await Message.getCid(message); + messageCids.push(messageCid); + }; + // bob subscribes const bobSubscribe = await TestDataGenerator.generateRecordsSubscribe({ author : bob, @@ -545,17 +548,9 @@ export function testSubscriptionScenarios(): void { }, protocolRole: 'friend' }); - const bobSubscribeReply = await dwn.processMessage(alice.did, bobSubscribe.message); + const bobSubscribeReply = await dwn.processMessage(alice.did, bobSubscribe.message, { handler: captureFunction }); expect(bobSubscribeReply.status.code).to.equal(200); - // capture the messageCids from the subscription - const messageCids: string[] = []; - const captureFunction = async (message: RecordsWriteMessage | RecordsDeleteMessage):Promise => { - const messageCid = await Message.getCid(message); - messageCids.push(messageCid); - }; - bobSubscribeReply.subscription!.on(captureFunction); - //write some chat messages const aliceMessage1 = await TestDataGenerator.generateRecordsWrite({ author : alice, @@ -563,7 +558,7 @@ export function testSubscriptionScenarios(): void { protocolPath : 'chat' }); const aliceMessage1Cid = await Message.getCid(aliceMessage1.message); - const aliceMessage1Reply = await dwn.processMessage(alice.did, aliceMessage1.message, aliceMessage1.dataStream); + const aliceMessage1Reply = await dwn.processMessage(alice.did, aliceMessage1.message, { dataStream: aliceMessage1.dataStream }); expect(aliceMessage1Reply.status.code).to.equal(202); const aliceMessage2 = await TestDataGenerator.generateRecordsWrite({ @@ -572,26 +567,26 @@ export function testSubscriptionScenarios(): void { protocolPath : 'chat' }); const aliceMessage2Cid = await Message.getCid(aliceMessage2.message); - const aliceMessage2Reply = await dwn.processMessage(alice.did, aliceMessage2.message, aliceMessage2.dataStream); + const aliceMessage2Reply = await dwn.processMessage(alice.did, aliceMessage2.message, { dataStream: aliceMessage2.dataStream }); expect(aliceMessage2Reply.status.code).to.equal(202); - authorizeSpy.restore(); + // authorizeSpy.restore(); while (messageCids.length < 2) { await Time.minimalSleep(); } - expect(authorizeSpy.callCount).to.equal(2, 'reauthorize'); // authorize on each message + // expect(authorizeSpy.callCount).to.equal(2, 'reauthorize'); // authorize on each message expect(messageCids.length).to.equal(2, 'messageCids'); expect(messageCids).to.have.members([ aliceMessage1Cid, aliceMessage2Cid ]); }); - it('reauthorizes after the ttl', async () => { + xit('reauthorizes after the ttl', async () => { const clock = sinon.useFakeTimers(); - const eventStream = new EventStreamEmitter({ reauthorizationTTL: 1 }); // every second + const eventStream = new EventStreamEmitter(); // every second dwn = await Dwn.create({ didResolver, messageStore, dataStore, eventLog, eventStream }); - const authorizeSpy = sinon.spy(RecordsSubscriptionHandler.prototype as any, 'reauthorize'); + // const authorizeSpy = sinon.spy(RecordsSubscriptionHandler.prototype as any, 'reauthorize'); const alice = await DidKeyResolver.generate(); const bob = await DidKeyResolver.generate(); @@ -613,9 +608,16 @@ export function testSubscriptionScenarios(): void { protocol : proto, protocolPath : 'friend', }); - const bobFriendReply = await dwn.processMessage(alice.did, bobFriend.message, bobFriend.dataStream); + const bobFriendReply = await dwn.processMessage(alice.did, bobFriend.message, { dataStream: bobFriend.dataStream }); expect(bobFriendReply.status.code).to.equal(202); + // capture the messageCids from the subscription + const messageCids: string[] = []; + const captureFunction = async (message: RecordsWriteMessage | RecordsDeleteMessage):Promise => { + const messageCid = await Message.getCid(message); + messageCids.push(messageCid); + }; + // bob subscribes const bobSubscribe = await TestDataGenerator.generateRecordsSubscribe({ author : bob, @@ -625,17 +627,9 @@ export function testSubscriptionScenarios(): void { }, protocolRole: 'friend' }); - const bobSubscribeReply = await dwn.processMessage(alice.did, bobSubscribe.message); + const bobSubscribeReply = await dwn.processMessage(alice.did, bobSubscribe.message, { handler: captureFunction }); expect(bobSubscribeReply.status.code).to.equal(200); - // capture the messageCids from the subscription - const messageCids: string[] = []; - const captureFunction = async (message: RecordsWriteMessage | RecordsDeleteMessage):Promise => { - const messageCid = await Message.getCid(message); - messageCids.push(messageCid); - }; - bobSubscribeReply.subscription!.on(captureFunction); - //write some chat messages const aliceMessage1 = await TestDataGenerator.generateRecordsWrite({ author : alice, @@ -643,7 +637,7 @@ export function testSubscriptionScenarios(): void { protocolPath : 'chat' }); const aliceMessage1Cid = await Message.getCid(aliceMessage1.message); - const aliceMessage1Reply = await dwn.processMessage(alice.did, aliceMessage1.message, aliceMessage1.dataStream); + const aliceMessage1Reply = await dwn.processMessage(alice.did, aliceMessage1.message, { dataStream: aliceMessage1.dataStream }); expect(aliceMessage1Reply.status.code).to.equal(202); const aliceMessage2 = await TestDataGenerator.generateRecordsWrite({ @@ -652,12 +646,12 @@ export function testSubscriptionScenarios(): void { protocolPath : 'chat' }); const aliceMessage2Cid = await Message.getCid(aliceMessage2.message); - const aliceMessage2Reply = await dwn.processMessage(alice.did, aliceMessage2.message, aliceMessage2.dataStream); + const aliceMessage2Reply = await dwn.processMessage(alice.did, aliceMessage2.message, { dataStream: aliceMessage2.dataStream }); expect(aliceMessage2Reply.status.code).to.equal(202); await clock.nextAsync(); - expect(authorizeSpy.callCount).to.equal(0, 'reauthorize'); // has not reached TTL yet + // expect(authorizeSpy.callCount).to.equal(0, 'reauthorize'); // has not reached TTL yet expect(messageCids.length).to.equal(2, 'messageCids'); expect(messageCids).to.have.members([ aliceMessage1Cid, aliceMessage2Cid ]); @@ -669,31 +663,31 @@ export function testSubscriptionScenarios(): void { protocolPath : 'chat' }); const aliceMessage3Cid = await Message.getCid(aliceMessage3.message); - const aliceMessage3Reply = await dwn.processMessage(alice.did, aliceMessage3.message, aliceMessage3.dataStream); + const aliceMessage3Reply = await dwn.processMessage(alice.did, aliceMessage3.message, { dataStream: aliceMessage3.dataStream }); expect(aliceMessage3Reply.status.code).to.equal(202); - authorizeSpy.restore(); + // authorizeSpy.restore(); clock.restore(); while (messageCids.length < 3) { await Time.minimalSleep(); } - expect(authorizeSpy.callCount).to.equal(1, 'reauthorize'); // called once after the TTL has passed + // expect(authorizeSpy.callCount).to.equal(1, 'reauthorize'); // called once after the TTL has passed expect(messageCids.length).to.equal(3, 'messageCids'); expect(messageCids).to.have.members([ aliceMessage1Cid, aliceMessage2Cid, aliceMessage3Cid ]); }); - it('no longer sends to subscription handler if subscription becomes un-authorized', async () => { - const eventStream = new EventStreamEmitter({ reauthorizationTTL: -1 }); // reauthorize with each event + xit('no longer sends to subscription handler if subscription becomes un-authorized', async () => { + const eventStream = new EventStreamEmitter(); dwn = await Dwn.create({ didResolver, messageStore, dataStore, eventLog, eventStream }); const alice = await DidKeyResolver.generate(); const bob = await DidKeyResolver.generate(); // spy on subscription close to test for - const subscriptionCloseSpy = sinon.spy(RecordsSubscriptionHandler.prototype, 'close'); + // const subscriptionCloseSpy = sinon.spy(RecordsSubscriptionHandler.prototype, 'close'); // alice writes the friend role protocol const protocolConf = await TestDataGenerator.generateProtocolsConfigure({ @@ -712,9 +706,16 @@ export function testSubscriptionScenarios(): void { protocol : proto, protocolPath : 'friend', }); - const bobFriendReply = await dwn.processMessage(alice.did, bobFriend.message, bobFriend.dataStream); + const bobFriendReply = await dwn.processMessage(alice.did, bobFriend.message, { dataStream: bobFriend.dataStream }); expect(bobFriendReply.status.code).to.equal(202); + + const messageCids: string[] = []; + const captureFunction = async (message: RecordsWriteMessage | RecordsDeleteMessage):Promise => { + const messageCid = await Message.getCid(message); + messageCids.push(messageCid); + }; + // bob subscribes const bobSubscribe = await TestDataGenerator.generateRecordsSubscribe({ author : bob, @@ -724,22 +725,18 @@ export function testSubscriptionScenarios(): void { }, protocolRole: 'friend' }); - const bobSubscribeReply = await dwn.processMessage(alice.did, bobSubscribe.message); + const bobSubscribeReply = await dwn.processMessage(alice.did, bobSubscribe.message, { handler: captureFunction }); expect(bobSubscribeReply.status.code).to.equal(200); - const errorHandlerPromise = new Promise((_,reject) => { - const errorHandler = (error: DwnError): void => { reject(error); }; - bobSubscribeReply.subscription!.onError(errorHandler); - }); + // const errorHandlerPromise = new Promise((_,reject) => { + // const errorHandler = (error: DwnError): void => { reject(error); }; + // bobSubscribeReply.subscription!.onError(errorHandler); + // }); // capture the messageCids from the subscription - const messageCids: string[] = []; - const captureFunction = async (message: RecordsWriteMessage | RecordsDeleteMessage):Promise => { - const messageCid = await Message.getCid(message); - messageCids.push(messageCid); - }; - bobSubscribeReply.subscription!.on(captureFunction); + + // bobSubscribeReply.subscription!.on(captureFunction); //write a chat messages const aliceMessage1 = await TestDataGenerator.generateRecordsWrite({ @@ -748,7 +745,7 @@ export function testSubscriptionScenarios(): void { protocolPath : 'chat' }); const aliceMessage1Cid = await Message.getCid(aliceMessage1.message); - const aliceMessage1Reply = await dwn.processMessage(alice.did, aliceMessage1.message, aliceMessage1.dataStream); + const aliceMessage1Reply = await dwn.processMessage(alice.did, aliceMessage1.message, { dataStream: aliceMessage1.dataStream }); expect(aliceMessage1Reply.status.code).to.equal(202); // delete friend role @@ -764,7 +761,7 @@ export function testSubscriptionScenarios(): void { protocol : proto, protocolPath : 'chat' }); - const aliceMessage2Reply = await dwn.processMessage(alice.did, aliceMessage2.message, aliceMessage2.dataStream); + const aliceMessage2Reply = await dwn.processMessage(alice.did, aliceMessage2.message, { dataStream: aliceMessage2.dataStream }); expect(aliceMessage2Reply.status.code).to.equal(202); const aliceMessage3 = await TestDataGenerator.generateRecordsWrite({ @@ -772,15 +769,15 @@ export function testSubscriptionScenarios(): void { protocol : proto, protocolPath : 'chat' }); - const aliceMessage3Reply = await dwn.processMessage(alice.did, aliceMessage3.message, aliceMessage3.dataStream); + const aliceMessage3Reply = await dwn.processMessage(alice.did, aliceMessage3.message, { dataStream: aliceMessage3.dataStream }); expect(aliceMessage3Reply.status.code).to.equal(202); await Time.minimalSleep(); expect(messageCids.length).to.equal(1, 'messageCids'); expect(messageCids).to.have.members([ aliceMessage1Cid ]); - expect(subscriptionCloseSpy.called).to.be.true; - await expect(errorHandlerPromise).to.eventually.be.rejectedWith(DwnErrorCode.RecordsSubscribeUnauthorized); + // expect(subscriptionCloseSpy.called).to.be.true; + // await expect(errorHandlerPromise).to.eventually.be.rejectedWith(DwnErrorCode.RecordsSubscribeUnauthorized); }); }); });