@webiny/api-headless-cms-scheduler 0.0.0-unstable.e53eceafb5 → 0.0.0-unstable.e6f0dc8ca7
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.
- package/README.md +11 -15
- package/context.d.ts +2 -7
- package/context.js +15 -59
- package/context.js.map +1 -1
- package/exports/api/cms/scheduler.d.ts +2 -0
- package/exports/api/cms/scheduler.js +4 -0
- package/exports/api/cms/scheduler.js.map +1 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnEntryDeleteEventHandler.d.ts +20 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnEntryDeleteEventHandler.js +54 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnEntryDeleteEventHandler.js.map +1 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnPublishEventHandler.d.ts +19 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnPublishEventHandler.js +49 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnPublishEventHandler.js.map +1 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnRevisionDeleteEventHandler.d.ts +19 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnRevisionDeleteEventHandler.js +48 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnRevisionDeleteEventHandler.js.map +1 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnUnpublishEventHandler.d.ts +19 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnUnpublishEventHandler.js +49 -0
- package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnUnpublishEventHandler.js.map +1 -0
- package/features/CancelScheduledActionOnEntryChange/feature.d.ts +11 -0
- package/features/CancelScheduledActionOnEntryChange/feature.js +24 -0
- package/features/CancelScheduledActionOnEntryChange/feature.js.map +1 -0
- package/features/NamespaceHandler/NamespaceHandler.d.ts +15 -0
- package/features/NamespaceHandler/NamespaceHandler.js +50 -0
- package/features/NamespaceHandler/NamespaceHandler.js.map +1 -0
- package/features/PublishActionHandler/PublishEntryActionHandler.d.ts +35 -0
- package/features/PublishActionHandler/PublishEntryActionHandler.js +114 -0
- package/features/PublishActionHandler/PublishEntryActionHandler.js.map +1 -0
- package/features/SchedulePublishEntryUseCase/SchedulePublishEntryUseCase.d.ts +15 -0
- package/features/SchedulePublishEntryUseCase/SchedulePublishEntryUseCase.js +51 -0
- package/features/SchedulePublishEntryUseCase/SchedulePublishEntryUseCase.js.map +1 -0
- package/features/SchedulePublishEntryUseCase/abstractions.d.ts +23 -0
- package/features/SchedulePublishEntryUseCase/abstractions.js +5 -0
- package/features/SchedulePublishEntryUseCase/abstractions.js.map +1 -0
- package/features/ScheduleUnpublishEntryUseCase/ScheduleUnpublishEntryUseCase.d.ts +15 -0
- package/features/ScheduleUnpublishEntryUseCase/ScheduleUnpublishEntryUseCase.js +51 -0
- package/features/ScheduleUnpublishEntryUseCase/ScheduleUnpublishEntryUseCase.js.map +1 -0
- package/features/ScheduleUnpublishEntryUseCase/abstractions.d.ts +23 -0
- package/features/ScheduleUnpublishEntryUseCase/abstractions.js +5 -0
- package/features/ScheduleUnpublishEntryUseCase/abstractions.js.map +1 -0
- package/features/UnpublishActionHandler/UnpublishEntryActionHandler.d.ts +30 -0
- package/features/UnpublishActionHandler/UnpublishEntryActionHandler.js +97 -0
- package/features/UnpublishActionHandler/UnpublishEntryActionHandler.js.map +1 -0
- package/index.d.ts +2 -10
- package/index.js +3 -26
- package/index.js.map +1 -1
- package/package.json +22 -23
- package/types.d.ts +3 -10
- package/types.js +1 -5
- package/types.js.map +1 -1
- package/utils/namespace.d.ts +4 -0
- package/utils/namespace.js +12 -0
- package/utils/namespace.js.map +1 -0
- package/constants.d.ts +0 -11
- package/constants.js +0 -19
- package/constants.js.map +0 -1
- package/graphql/index.d.ts +0 -3
- package/graphql/index.js +0 -204
- package/graphql/index.js.map +0 -1
- package/graphql/schema.d.ts +0 -217
- package/graphql/schema.js +0 -87
- package/graphql/schema.js.map +0 -1
- package/handler/Handler.d.ts +0 -23
- package/handler/Handler.js +0 -74
- package/handler/Handler.js.map +0 -1
- package/handler/actions/PublishHandlerAction.d.ts +0 -13
- package/handler/actions/PublishHandlerAction.js +0 -64
- package/handler/actions/PublishHandlerAction.js.map +0 -1
- package/handler/actions/UnpublishHandlerAction.d.ts +0 -13
- package/handler/actions/UnpublishHandlerAction.js +0 -53
- package/handler/actions/UnpublishHandlerAction.js.map +0 -1
- package/handler/index.d.ts +0 -7
- package/handler/index.js +0 -64
- package/handler/index.js.map +0 -1
- package/handler/types.d.ts +0 -5
- package/handler/types.js +0 -7
- package/handler/types.js.map +0 -1
- package/manifest.d.ts +0 -17
- package/manifest.js +0 -47
- package/manifest.js.map +0 -1
- package/scheduler/ScheduleExecutor.d.ts +0 -16
- package/scheduler/ScheduleExecutor.js +0 -55
- package/scheduler/ScheduleExecutor.js.map +0 -1
- package/scheduler/ScheduleFetcher.d.ts +0 -16
- package/scheduler/ScheduleFetcher.js +0 -51
- package/scheduler/ScheduleFetcher.js.map +0 -1
- package/scheduler/ScheduleRecord.d.ts +0 -33
- package/scheduler/ScheduleRecord.js +0 -56
- package/scheduler/ScheduleRecord.js.map +0 -1
- package/scheduler/Scheduler.d.ts +0 -14
- package/scheduler/Scheduler.js +0 -27
- package/scheduler/Scheduler.js.map +0 -1
- package/scheduler/actions/PublishScheduleAction.d.ts +0 -26
- package/scheduler/actions/PublishScheduleAction.js +0 -209
- package/scheduler/actions/PublishScheduleAction.js.map +0 -1
- package/scheduler/actions/UnpublishScheduleAction.d.ts +0 -26
- package/scheduler/actions/UnpublishScheduleAction.js +0 -199
- package/scheduler/actions/UnpublishScheduleAction.js.map +0 -1
- package/scheduler/createScheduleRecordId.d.ts +0 -2
- package/scheduler/createScheduleRecordId.js +0 -33
- package/scheduler/createScheduleRecordId.js.map +0 -1
- package/scheduler/createScheduler.d.ts +0 -12
- package/scheduler/createScheduler.js +0 -59
- package/scheduler/createScheduler.js.map +0 -1
- package/scheduler/dates.d.ts +0 -7
- package/scheduler/dates.js +0 -27
- package/scheduler/dates.js.map +0 -1
- package/scheduler/model.d.ts +0 -1
- package/scheduler/model.js +0 -88
- package/scheduler/model.js.map +0 -1
- package/scheduler/types.d.ts +0 -92
- package/scheduler/types.js +0 -19
- package/scheduler/types.js.map +0 -1
- package/service/SchedulerService.d.ts +0 -23
- package/service/SchedulerService.js +0 -151
- package/service/SchedulerService.js.map +0 -1
- package/service/types.d.ts +0 -19
- package/service/types.js +0 -7
- package/service/types.js.map +0 -1
- package/utils/dateInTheFuture.d.ts +0 -6
- package/utils/dateInTheFuture.js +0 -19
- package/utils/dateInTheFuture.js.map +0 -1
package/README.md
CHANGED
|
@@ -1,15 +1,11 @@
|
|
|
1
|
-
# @webiny/api-headless-cms-
|
|
2
|
-
|
|
3
|
-
[!
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
Or if you prefer yarn:
|
|
13
|
-
```
|
|
14
|
-
yarn add @webiny/api-headless-cms-schedule
|
|
15
|
-
```
|
|
1
|
+
# @webiny/api-headless-cms-scheduler
|
|
2
|
+
|
|
3
|
+
> [!NOTE]
|
|
4
|
+
> This package is part of the [Webiny](https://www.webiny.com) monorepo.
|
|
5
|
+
> It’s **included in every Webiny project by default** and is not meant to be used as a standalone package.
|
|
6
|
+
|
|
7
|
+
📘 **Documentation:** [https://www.webiny.com/docs](https://www.webiny.com/docs)
|
|
8
|
+
|
|
9
|
+
---
|
|
10
|
+
|
|
11
|
+
_This README file is automatically generated during the publish process._
|
package/context.d.ts
CHANGED
|
@@ -1,7 +1,2 @@
|
|
|
1
|
-
import { ContextPlugin } from "@webiny/api
|
|
2
|
-
|
|
3
|
-
import type { ScheduleContext } from "./types.js";
|
|
4
|
-
export interface ICreateHeadlessCmsSchedulerContextParams {
|
|
5
|
-
getClient(config?: SchedulerClientConfig): Pick<SchedulerClient, "send">;
|
|
6
|
-
}
|
|
7
|
-
export declare const createHeadlessCmsScheduleContext: (params: ICreateHeadlessCmsSchedulerContextParams) => ContextPlugin<ScheduleContext>;
|
|
1
|
+
import { ContextPlugin } from "@webiny/api";
|
|
2
|
+
export declare const createHeadlessCmsScheduleContext: () => ContextPlugin<import("@webiny/api/types").Context>;
|
package/context.js
CHANGED
|
@@ -1,63 +1,19 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
}
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
return new _index.ContextPlugin(async context => {
|
|
17
|
-
/**
|
|
18
|
-
* If the Headless CMS is not ready, it means the system is not fully installed yet.
|
|
19
|
-
* We do not want to continue because it would break anyway.
|
|
20
|
-
*/
|
|
21
|
-
const ready = await (0, _apiHeadlessCms.isHeadlessCmsReady)(context);
|
|
22
|
-
if (!ready) {
|
|
23
|
-
return;
|
|
24
|
-
}
|
|
25
|
-
const manifest = await (0, _manifest.getManifest)({
|
|
26
|
-
client: context.db.driver.getClient()
|
|
27
|
-
});
|
|
28
|
-
if (manifest.error) {
|
|
29
|
-
console.error(manifest.error.message);
|
|
30
|
-
console.log((0, _utils.convertException)(manifest.error, ["message"]));
|
|
31
|
-
console.info("Scheduler not attached.");
|
|
32
|
-
return;
|
|
33
|
-
}
|
|
34
|
-
const service = (0, _SchedulerService.createSchedulerService)({
|
|
35
|
-
getClient: params.getClient,
|
|
36
|
-
config: {
|
|
37
|
-
lambdaArn: manifest.data.lambdaArn,
|
|
38
|
-
roleArn: manifest.data.roleArn
|
|
39
|
-
}
|
|
40
|
-
});
|
|
41
|
-
let scheduleModel;
|
|
42
|
-
try {
|
|
43
|
-
scheduleModel = await context.cms.getModel(_constants.SCHEDULE_MODEL_ID);
|
|
44
|
-
} catch (ex) {
|
|
45
|
-
if (ex.code === "NOT_FOUND" || ex instanceof _handlerGraphql.NotFoundError) {
|
|
46
|
-
console.error(`Schedule model "${_constants.SCHEDULE_MODEL_ID}" not found.`);
|
|
47
|
-
return;
|
|
48
|
-
}
|
|
49
|
-
console.error(`Error while fetching schedule model "${_constants.SCHEDULE_MODEL_ID}".`);
|
|
50
|
-
console.log((0, _utils.convertException)(ex));
|
|
51
|
-
return;
|
|
52
|
-
}
|
|
53
|
-
context.cms.scheduler = await (0, _createScheduler.createScheduler)({
|
|
54
|
-
cms: context.cms,
|
|
55
|
-
security: context.security,
|
|
56
|
-
service,
|
|
57
|
-
scheduleModel
|
|
58
|
-
});
|
|
1
|
+
import { ContextPlugin } from "@webiny/api";
|
|
2
|
+
import { CancelScheduledActionOnEntryChangeFeature } from "./features/CancelScheduledActionOnEntryChange/feature.js";
|
|
3
|
+
import { NamespaceHandler } from "./features/NamespaceHandler/NamespaceHandler.js";
|
|
4
|
+
import { PublishEntryActionHandler } from "./features/PublishActionHandler/PublishEntryActionHandler.js";
|
|
5
|
+
import { UnpublishEntryActionHandler } from "./features/UnpublishActionHandler/UnpublishEntryActionHandler.js";
|
|
6
|
+
import { SchedulePublishEntryUseCase } from "./features/SchedulePublishEntryUseCase/SchedulePublishEntryUseCase.js";
|
|
7
|
+
import { ScheduleUnpublishEntryUseCase } from "./features/ScheduleUnpublishEntryUseCase/ScheduleUnpublishEntryUseCase.js";
|
|
8
|
+
export const createHeadlessCmsScheduleContext = () => {
|
|
9
|
+
return new ContextPlugin(async context => {
|
|
10
|
+
context.container.register(NamespaceHandler);
|
|
11
|
+
context.container.register(PublishEntryActionHandler);
|
|
12
|
+
context.container.register(UnpublishEntryActionHandler);
|
|
13
|
+
context.container.register(SchedulePublishEntryUseCase);
|
|
14
|
+
context.container.register(ScheduleUnpublishEntryUseCase);
|
|
15
|
+
CancelScheduledActionOnEntryChangeFeature.register(context.container);
|
|
59
16
|
});
|
|
60
17
|
};
|
|
61
|
-
exports.createHeadlessCmsScheduleContext = createHeadlessCmsScheduleContext;
|
|
62
18
|
|
|
63
19
|
//# sourceMappingURL=context.js.map
|
package/context.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"names":["
|
|
1
|
+
{"version":3,"names":["ContextPlugin","CancelScheduledActionOnEntryChangeFeature","NamespaceHandler","PublishEntryActionHandler","UnpublishEntryActionHandler","SchedulePublishEntryUseCase","ScheduleUnpublishEntryUseCase","createHeadlessCmsScheduleContext","context","container","register"],"sources":["context.ts"],"sourcesContent":["import { ContextPlugin } from \"@webiny/api\";\nimport { CancelScheduledActionOnEntryChangeFeature } from \"~/features/CancelScheduledActionOnEntryChange/feature.js\";\nimport { NamespaceHandler } from \"~/features/NamespaceHandler/NamespaceHandler.js\";\nimport { PublishEntryActionHandler } from \"~/features/PublishActionHandler/PublishEntryActionHandler.js\";\nimport { UnpublishEntryActionHandler } from \"~/features/UnpublishActionHandler/UnpublishEntryActionHandler.js\";\nimport { SchedulePublishEntryUseCase } from \"~/features/SchedulePublishEntryUseCase/SchedulePublishEntryUseCase.js\";\nimport { ScheduleUnpublishEntryUseCase } from \"~/features/ScheduleUnpublishEntryUseCase/ScheduleUnpublishEntryUseCase.js\";\n\nexport const createHeadlessCmsScheduleContext = () => {\n return new ContextPlugin(async context => {\n context.container.register(NamespaceHandler);\n context.container.register(PublishEntryActionHandler);\n context.container.register(UnpublishEntryActionHandler);\n context.container.register(SchedulePublishEntryUseCase);\n context.container.register(ScheduleUnpublishEntryUseCase);\n\n CancelScheduledActionOnEntryChangeFeature.register(context.container);\n });\n};\n"],"mappings":"AAAA,SAASA,aAAa,QAAQ,aAAa;AAC3C,SAASC,yCAAyC;AAClD,SAASC,gBAAgB;AACzB,SAASC,yBAAyB;AAClC,SAASC,2BAA2B;AACpC,SAASC,2BAA2B;AACpC,SAASC,6BAA6B;AAEtC,OAAO,MAAMC,gCAAgC,GAAGA,CAAA,KAAM;EAClD,OAAO,IAAIP,aAAa,CAAC,MAAMQ,OAAO,IAAI;IACtCA,OAAO,CAACC,SAAS,CAACC,QAAQ,CAACR,gBAAgB,CAAC;IAC5CM,OAAO,CAACC,SAAS,CAACC,QAAQ,CAACP,yBAAyB,CAAC;IACrDK,OAAO,CAACC,SAAS,CAACC,QAAQ,CAACN,2BAA2B,CAAC;IACvDI,OAAO,CAACC,SAAS,CAACC,QAAQ,CAACL,2BAA2B,CAAC;IACvDG,OAAO,CAACC,SAAS,CAACC,QAAQ,CAACJ,6BAA6B,CAAC;IAEzDL,yCAAyC,CAACS,QAAQ,CAACF,OAAO,CAACC,SAAS,CAAC;EACzE,CAAC,CAAC;AACN,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["SchedulePublishEntryUseCase","ScheduleUnpublishEntryUseCase"],"sources":["scheduler.ts"],"sourcesContent":["export { SchedulePublishEntryUseCase } from \"~/features/SchedulePublishEntryUseCase/abstractions.js\";\nexport { ScheduleUnpublishEntryUseCase } from \"~/features/ScheduleUnpublishEntryUseCase/abstractions.js\";\n"],"mappings":"AAAA,SAASA,2BAA2B;AACpC,SAASC,6BAA6B","ignoreList":[]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { EntryAfterDeleteEventHandler } from "@webiny/api-headless-cms/features/contentEntry/DeleteEntry/events";
|
|
2
|
+
import { CancelScheduledActionUseCase, ListScheduledActionsUseCase } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
3
|
+
/**
|
|
4
|
+
* Cancels scheduled actions when an entry is deleted
|
|
5
|
+
*
|
|
6
|
+
* When a user deletes an entry, any scheduled publish/unpublish
|
|
7
|
+
* actions for all of its revisions should be canceled since the entry
|
|
8
|
+
* no longer exists.
|
|
9
|
+
*/
|
|
10
|
+
declare class CancelScheduledActionOnEntryDeleteHandlerImpl implements EntryAfterDeleteEventHandler.Interface {
|
|
11
|
+
private listScheduledActions;
|
|
12
|
+
private cancelScheduledAction;
|
|
13
|
+
constructor(listScheduledActions: ListScheduledActionsUseCase.Interface, cancelScheduledAction: CancelScheduledActionUseCase.Interface);
|
|
14
|
+
handle(event: EntryAfterDeleteEventHandler.Event): Promise<void>;
|
|
15
|
+
private listSchedules;
|
|
16
|
+
}
|
|
17
|
+
export declare const CancelScheduledActionOnEntryDeleteEventHandler: typeof CancelScheduledActionOnEntryDeleteHandlerImpl & {
|
|
18
|
+
__abstraction: import("@webiny/di").Abstraction<import("@webiny/api-core/features/eventPublisher").IEventHandler<import("@webiny/api-headless-cms/features/contentEntry/DeleteEntry/events").EntryAfterDeleteEvent>>;
|
|
19
|
+
};
|
|
20
|
+
export {};
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { EntryAfterDeleteEventHandler } from "@webiny/api-headless-cms/features/contentEntry/DeleteEntry/events";
|
|
2
|
+
import { CancelScheduledActionUseCase, ListScheduledActionsUseCase } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
3
|
+
import { createNamespace } from "../../utils/namespace.js";
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Cancels scheduled actions when an entry is deleted
|
|
7
|
+
*
|
|
8
|
+
* When a user deletes an entry, any scheduled publish/unpublish
|
|
9
|
+
* actions for all of its revisions should be canceled since the entry
|
|
10
|
+
* no longer exists.
|
|
11
|
+
*/
|
|
12
|
+
class CancelScheduledActionOnEntryDeleteHandlerImpl {
|
|
13
|
+
constructor(listScheduledActions, cancelScheduledAction) {
|
|
14
|
+
this.listScheduledActions = listScheduledActions;
|
|
15
|
+
this.cancelScheduledAction = cancelScheduledAction;
|
|
16
|
+
}
|
|
17
|
+
async handle(event) {
|
|
18
|
+
const {
|
|
19
|
+
entry,
|
|
20
|
+
model
|
|
21
|
+
} = event.payload;
|
|
22
|
+
|
|
23
|
+
// Skip private models
|
|
24
|
+
if (model.isPrivate) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
const schedules = await this.listSchedules(model.modelId, entry.entryId);
|
|
28
|
+
for (const action of schedules) {
|
|
29
|
+
const cancelRes = await this.cancelScheduledAction.execute(action);
|
|
30
|
+
if (cancelRes.isFail()) {
|
|
31
|
+
// Silently ignore errors - this is non-critical cleanup.
|
|
32
|
+
// The entry was deleted successfully, cancelling scheduled actions is best-effort.
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
async listSchedules(modelId, entryId) {
|
|
37
|
+
const actionsResult = await this.listScheduledActions.execute({
|
|
38
|
+
limit: 10000,
|
|
39
|
+
where: {
|
|
40
|
+
namespace: createNamespace({
|
|
41
|
+
modelId
|
|
42
|
+
}),
|
|
43
|
+
targetId_startsWith: `${entryId}#`
|
|
44
|
+
}
|
|
45
|
+
});
|
|
46
|
+
return actionsResult.value.items;
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
export const CancelScheduledActionOnEntryDeleteEventHandler = EntryAfterDeleteEventHandler.createImplementation({
|
|
50
|
+
implementation: CancelScheduledActionOnEntryDeleteHandlerImpl,
|
|
51
|
+
dependencies: [ListScheduledActionsUseCase, CancelScheduledActionUseCase]
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
//# sourceMappingURL=CancelScheduledActionOnEntryDeleteEventHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["EntryAfterDeleteEventHandler","CancelScheduledActionUseCase","ListScheduledActionsUseCase","createNamespace","CancelScheduledActionOnEntryDeleteHandlerImpl","constructor","listScheduledActions","cancelScheduledAction","handle","event","entry","model","payload","isPrivate","schedules","listSchedules","modelId","entryId","action","cancelRes","execute","isFail","actionsResult","limit","where","namespace","targetId_startsWith","value","items","CancelScheduledActionOnEntryDeleteEventHandler","createImplementation","implementation","dependencies"],"sources":["CancelScheduledActionOnEntryDeleteEventHandler.ts"],"sourcesContent":["import { EntryAfterDeleteEventHandler } from \"@webiny/api-headless-cms/features/contentEntry/DeleteEntry/events\";\nimport {\n CancelScheduledActionUseCase,\n ListScheduledActionsUseCase\n} from \"@webiny/api-scheduler/exports/api/scheduler.js\";\nimport { createNamespace } from \"~/utils/namespace.js\";\n\n/**\n * Cancels scheduled actions when an entry is deleted\n *\n * When a user deletes an entry, any scheduled publish/unpublish\n * actions for all of its revisions should be canceled since the entry\n * no longer exists.\n */\nclass CancelScheduledActionOnEntryDeleteHandlerImpl\n implements EntryAfterDeleteEventHandler.Interface\n{\n constructor(\n private listScheduledActions: ListScheduledActionsUseCase.Interface,\n private cancelScheduledAction: CancelScheduledActionUseCase.Interface\n ) {}\n\n async handle(event: EntryAfterDeleteEventHandler.Event): Promise<void> {\n const { entry, model } = event.payload;\n\n // Skip private models\n if (model.isPrivate) {\n return;\n }\n\n const schedules = await this.listSchedules(model.modelId, entry.entryId);\n for (const action of schedules) {\n const cancelRes = await this.cancelScheduledAction.execute(action);\n if (cancelRes.isFail()) {\n // Silently ignore errors - this is non-critical cleanup.\n // The entry was deleted successfully, cancelling scheduled actions is best-effort.\n }\n }\n }\n\n private async listSchedules(modelId: string, entryId: string) {\n const actionsResult = await this.listScheduledActions.execute({\n limit: 10000,\n where: {\n namespace: createNamespace({ modelId }),\n targetId_startsWith: `${entryId}#`\n }\n });\n\n return actionsResult.value.items;\n }\n}\n\nexport const CancelScheduledActionOnEntryDeleteEventHandler =\n EntryAfterDeleteEventHandler.createImplementation({\n implementation: CancelScheduledActionOnEntryDeleteHandlerImpl,\n dependencies: [ListScheduledActionsUseCase, CancelScheduledActionUseCase]\n });\n"],"mappings":"AAAA,SAASA,4BAA4B,QAAQ,mEAAmE;AAChH,SACIC,4BAA4B,EAC5BC,2BAA2B,QACxB,gDAAgD;AACvD,SAASC,eAAe;;AAExB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAMC,6CAA6C,CAEnD;EACIC,WAAWA,CACCC,oBAA2D,EAC3DC,qBAA6D,EACvE;IAAA,KAFUD,oBAA2D,GAA3DA,oBAA2D;IAAA,KAC3DC,qBAA6D,GAA7DA,qBAA6D;EACtE;EAEH,MAAMC,MAAMA,CAACC,KAAyC,EAAiB;IACnE,MAAM;MAAEC,KAAK;MAAEC;IAAM,CAAC,GAAGF,KAAK,CAACG,OAAO;;IAEtC;IACA,IAAID,KAAK,CAACE,SAAS,EAAE;MACjB;IACJ;IAEA,MAAMC,SAAS,GAAG,MAAM,IAAI,CAACC,aAAa,CAACJ,KAAK,CAACK,OAAO,EAAEN,KAAK,CAACO,OAAO,CAAC;IACxE,KAAK,MAAMC,MAAM,IAAIJ,SAAS,EAAE;MAC5B,MAAMK,SAAS,GAAG,MAAM,IAAI,CAACZ,qBAAqB,CAACa,OAAO,CAACF,MAAM,CAAC;MAClE,IAAIC,SAAS,CAACE,MAAM,CAAC,CAAC,EAAE;QACpB;QACA;MAAA;IAER;EACJ;EAEA,MAAcN,aAAaA,CAACC,OAAe,EAAEC,OAAe,EAAE;IAC1D,MAAMK,aAAa,GAAG,MAAM,IAAI,CAAChB,oBAAoB,CAACc,OAAO,CAAC;MAC1DG,KAAK,EAAE,KAAK;MACZC,KAAK,EAAE;QACHC,SAAS,EAAEtB,eAAe,CAAC;UAAEa;QAAQ,CAAC,CAAC;QACvCU,mBAAmB,EAAE,GAAGT,OAAO;MACnC;IACJ,CAAC,CAAC;IAEF,OAAOK,aAAa,CAACK,KAAK,CAACC,KAAK;EACpC;AACJ;AAEA,OAAO,MAAMC,8CAA8C,GACvD7B,4BAA4B,CAAC8B,oBAAoB,CAAC;EAC9CC,cAAc,EAAE3B,6CAA6C;EAC7D4B,YAAY,EAAE,CAAC9B,2BAA2B,EAAED,4BAA4B;AAC5E,CAAC,CAAC","ignoreList":[]}
|
package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnPublishEventHandler.d.ts
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { EntryAfterPublishEventHandler } from "@webiny/api-headless-cms/features/contentEntry/PublishEntry/events";
|
|
2
|
+
import { CancelScheduledActionUseCase, ListScheduledActionsUseCase } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
3
|
+
/**
|
|
4
|
+
* Cancels scheduled "publish" when an entry is manually published
|
|
5
|
+
*
|
|
6
|
+
* When a user manually publishes an entry, any scheduled publish
|
|
7
|
+
* action for that entry should be canceled since the manual action
|
|
8
|
+
* takes precedence.
|
|
9
|
+
*/
|
|
10
|
+
declare class CancelScheduledActionOnPublishEventHandlerImpl implements EntryAfterPublishEventHandler.Interface {
|
|
11
|
+
private listScheduledActions;
|
|
12
|
+
private cancelScheduledAction;
|
|
13
|
+
constructor(listScheduledActions: ListScheduledActionsUseCase.Interface, cancelScheduledAction: CancelScheduledActionUseCase.Interface);
|
|
14
|
+
handle(event: EntryAfterPublishEventHandler.Event): Promise<void>;
|
|
15
|
+
}
|
|
16
|
+
export declare const CancelScheduledActionOnPublishEventHandler: typeof CancelScheduledActionOnPublishEventHandlerImpl & {
|
|
17
|
+
__abstraction: import("@webiny/di").Abstraction<import("@webiny/api-core/features/eventPublisher").IEventHandler<import("@webiny/api-headless-cms/features/contentEntry/PublishEntry/events").EntryAfterPublishEvent>>;
|
|
18
|
+
};
|
|
19
|
+
export {};
|
package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnPublishEventHandler.js
ADDED
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { EntryAfterPublishEventHandler } from "@webiny/api-headless-cms/features/contentEntry/PublishEntry/events";
|
|
2
|
+
import { CancelScheduledActionUseCase, ListScheduledActionsUseCase, ScheduledActionTypePublish } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
3
|
+
import { createNamespace } from "../../utils/namespace.js";
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Cancels scheduled "publish" when an entry is manually published
|
|
7
|
+
*
|
|
8
|
+
* When a user manually publishes an entry, any scheduled publish
|
|
9
|
+
* action for that entry should be canceled since the manual action
|
|
10
|
+
* takes precedence.
|
|
11
|
+
*/
|
|
12
|
+
class CancelScheduledActionOnPublishEventHandlerImpl {
|
|
13
|
+
constructor(listScheduledActions, cancelScheduledAction) {
|
|
14
|
+
this.listScheduledActions = listScheduledActions;
|
|
15
|
+
this.cancelScheduledAction = cancelScheduledAction;
|
|
16
|
+
}
|
|
17
|
+
async handle(event) {
|
|
18
|
+
const {
|
|
19
|
+
entry,
|
|
20
|
+
model
|
|
21
|
+
} = event.payload;
|
|
22
|
+
|
|
23
|
+
// Skip private models
|
|
24
|
+
if (model.isPrivate) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
const actionsResult = await this.listScheduledActions.execute({
|
|
28
|
+
where: {
|
|
29
|
+
namespace: createNamespace(model),
|
|
30
|
+
actionType: ScheduledActionTypePublish,
|
|
31
|
+
targetId: entry.id
|
|
32
|
+
}
|
|
33
|
+
});
|
|
34
|
+
const actions = actionsResult.value.items;
|
|
35
|
+
for (const action of actions) {
|
|
36
|
+
const cancelRes = await this.cancelScheduledAction.execute(action);
|
|
37
|
+
if (cancelRes.isFail()) {
|
|
38
|
+
// Silently ignore errors - this is non-critical cleanup.
|
|
39
|
+
// Even if a schedule runs on an already published action, nothing bad will happen.
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
export const CancelScheduledActionOnPublishEventHandler = EntryAfterPublishEventHandler.createImplementation({
|
|
45
|
+
implementation: CancelScheduledActionOnPublishEventHandlerImpl,
|
|
46
|
+
dependencies: [ListScheduledActionsUseCase, CancelScheduledActionUseCase]
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
//# sourceMappingURL=CancelScheduledActionOnPublishEventHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["EntryAfterPublishEventHandler","CancelScheduledActionUseCase","ListScheduledActionsUseCase","ScheduledActionTypePublish","createNamespace","CancelScheduledActionOnPublishEventHandlerImpl","constructor","listScheduledActions","cancelScheduledAction","handle","event","entry","model","payload","isPrivate","actionsResult","execute","where","namespace","actionType","targetId","id","actions","value","items","action","cancelRes","isFail","CancelScheduledActionOnPublishEventHandler","createImplementation","implementation","dependencies"],"sources":["CancelScheduledActionOnPublishEventHandler.ts"],"sourcesContent":["import { EntryAfterPublishEventHandler } from \"@webiny/api-headless-cms/features/contentEntry/PublishEntry/events\";\nimport {\n CancelScheduledActionUseCase,\n ListScheduledActionsUseCase,\n ScheduledActionTypePublish\n} from \"@webiny/api-scheduler/exports/api/scheduler.js\";\nimport { createNamespace } from \"~/utils/namespace.js\";\n\n/**\n * Cancels scheduled \"publish\" when an entry is manually published\n *\n * When a user manually publishes an entry, any scheduled publish\n * action for that entry should be canceled since the manual action\n * takes precedence.\n */\nclass CancelScheduledActionOnPublishEventHandlerImpl\n implements EntryAfterPublishEventHandler.Interface\n{\n constructor(\n private listScheduledActions: ListScheduledActionsUseCase.Interface,\n private cancelScheduledAction: CancelScheduledActionUseCase.Interface\n ) {}\n\n async handle(event: EntryAfterPublishEventHandler.Event): Promise<void> {\n const { entry, model } = event.payload;\n\n // Skip private models\n if (model.isPrivate) {\n return;\n }\n\n const actionsResult = await this.listScheduledActions.execute({\n where: {\n namespace: createNamespace(model),\n actionType: ScheduledActionTypePublish,\n targetId: entry.id\n }\n });\n\n const actions = actionsResult.value.items;\n\n for (const action of actions) {\n const cancelRes = await this.cancelScheduledAction.execute(action);\n if (cancelRes.isFail()) {\n // Silently ignore errors - this is non-critical cleanup.\n // Even if a schedule runs on an already published action, nothing bad will happen.\n }\n }\n }\n}\n\nexport const CancelScheduledActionOnPublishEventHandler =\n EntryAfterPublishEventHandler.createImplementation({\n implementation: CancelScheduledActionOnPublishEventHandlerImpl,\n dependencies: [ListScheduledActionsUseCase, CancelScheduledActionUseCase]\n });\n"],"mappings":"AAAA,SAASA,6BAA6B,QAAQ,oEAAoE;AAClH,SACIC,4BAA4B,EAC5BC,2BAA2B,EAC3BC,0BAA0B,QACvB,gDAAgD;AACvD,SAASC,eAAe;;AAExB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAMC,8CAA8C,CAEpD;EACIC,WAAWA,CACCC,oBAA2D,EAC3DC,qBAA6D,EACvE;IAAA,KAFUD,oBAA2D,GAA3DA,oBAA2D;IAAA,KAC3DC,qBAA6D,GAA7DA,qBAA6D;EACtE;EAEH,MAAMC,MAAMA,CAACC,KAA0C,EAAiB;IACpE,MAAM;MAAEC,KAAK;MAAEC;IAAM,CAAC,GAAGF,KAAK,CAACG,OAAO;;IAEtC;IACA,IAAID,KAAK,CAACE,SAAS,EAAE;MACjB;IACJ;IAEA,MAAMC,aAAa,GAAG,MAAM,IAAI,CAACR,oBAAoB,CAACS,OAAO,CAAC;MAC1DC,KAAK,EAAE;QACHC,SAAS,EAAEd,eAAe,CAACQ,KAAK,CAAC;QACjCO,UAAU,EAAEhB,0BAA0B;QACtCiB,QAAQ,EAAET,KAAK,CAACU;MACpB;IACJ,CAAC,CAAC;IAEF,MAAMC,OAAO,GAAGP,aAAa,CAACQ,KAAK,CAACC,KAAK;IAEzC,KAAK,MAAMC,MAAM,IAAIH,OAAO,EAAE;MAC1B,MAAMI,SAAS,GAAG,MAAM,IAAI,CAAClB,qBAAqB,CAACQ,OAAO,CAACS,MAAM,CAAC;MAClE,IAAIC,SAAS,CAACC,MAAM,CAAC,CAAC,EAAE;QACpB;QACA;MAAA;IAER;EACJ;AACJ;AAEA,OAAO,MAAMC,0CAA0C,GACnD5B,6BAA6B,CAAC6B,oBAAoB,CAAC;EAC/CC,cAAc,EAAEzB,8CAA8C;EAC9D0B,YAAY,EAAE,CAAC7B,2BAA2B,EAAED,4BAA4B;AAC5E,CAAC,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { EntryAfterDeleteEventHandler } from "@webiny/api-headless-cms/features/contentEntry/DeleteEntry/events";
|
|
2
|
+
import { CancelScheduledActionUseCase, ListScheduledActionsUseCase } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
3
|
+
/**
|
|
4
|
+
* Cancels scheduled actions when an entry revision is deleted
|
|
5
|
+
*
|
|
6
|
+
* When a user deletes an entry revision, any scheduled publish/unpublish
|
|
7
|
+
* action for that revision should be canceled since the revision
|
|
8
|
+
* no longer exists.
|
|
9
|
+
*/
|
|
10
|
+
declare class CancelScheduledActionOnDeleteHandlerImpl implements EntryAfterDeleteEventHandler.Interface {
|
|
11
|
+
private listScheduledActions;
|
|
12
|
+
private cancelScheduledAction;
|
|
13
|
+
constructor(listScheduledActions: ListScheduledActionsUseCase.Interface, cancelScheduledAction: CancelScheduledActionUseCase.Interface);
|
|
14
|
+
handle(event: EntryAfterDeleteEventHandler.Event): Promise<void>;
|
|
15
|
+
}
|
|
16
|
+
export declare const CancelScheduledActionOnRevisionDeleteEventHandler: typeof CancelScheduledActionOnDeleteHandlerImpl & {
|
|
17
|
+
__abstraction: import("@webiny/di").Abstraction<import("@webiny/api-core/features/eventPublisher").IEventHandler<import("@webiny/api-headless-cms/features/contentEntry/DeleteEntry/events").EntryAfterDeleteEvent>>;
|
|
18
|
+
};
|
|
19
|
+
export {};
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { EntryAfterDeleteEventHandler } from "@webiny/api-headless-cms/features/contentEntry/DeleteEntry/events";
|
|
2
|
+
import { CancelScheduledActionUseCase, ListScheduledActionsUseCase } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
3
|
+
import { createNamespace } from "../../utils/namespace.js";
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Cancels scheduled actions when an entry revision is deleted
|
|
7
|
+
*
|
|
8
|
+
* When a user deletes an entry revision, any scheduled publish/unpublish
|
|
9
|
+
* action for that revision should be canceled since the revision
|
|
10
|
+
* no longer exists.
|
|
11
|
+
*/
|
|
12
|
+
class CancelScheduledActionOnDeleteHandlerImpl {
|
|
13
|
+
constructor(listScheduledActions, cancelScheduledAction) {
|
|
14
|
+
this.listScheduledActions = listScheduledActions;
|
|
15
|
+
this.cancelScheduledAction = cancelScheduledAction;
|
|
16
|
+
}
|
|
17
|
+
async handle(event) {
|
|
18
|
+
const {
|
|
19
|
+
entry,
|
|
20
|
+
model
|
|
21
|
+
} = event.payload;
|
|
22
|
+
|
|
23
|
+
// Skip private models
|
|
24
|
+
if (model.isPrivate) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
const actionsResult = await this.listScheduledActions.execute({
|
|
28
|
+
where: {
|
|
29
|
+
namespace: createNamespace(model),
|
|
30
|
+
targetId: entry.id
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
const actions = actionsResult.value.items;
|
|
34
|
+
for (const action of actions) {
|
|
35
|
+
const cancelRes = await this.cancelScheduledAction.execute(action);
|
|
36
|
+
if (cancelRes.isFail()) {
|
|
37
|
+
// Silently ignore errors - this is non-critical cleanup.
|
|
38
|
+
// Entry was deleted successfully, cancelling scheduled actions is best-effort.
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
export const CancelScheduledActionOnRevisionDeleteEventHandler = EntryAfterDeleteEventHandler.createImplementation({
|
|
44
|
+
implementation: CancelScheduledActionOnDeleteHandlerImpl,
|
|
45
|
+
dependencies: [ListScheduledActionsUseCase, CancelScheduledActionUseCase]
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
//# sourceMappingURL=CancelScheduledActionOnRevisionDeleteEventHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["EntryAfterDeleteEventHandler","CancelScheduledActionUseCase","ListScheduledActionsUseCase","createNamespace","CancelScheduledActionOnDeleteHandlerImpl","constructor","listScheduledActions","cancelScheduledAction","handle","event","entry","model","payload","isPrivate","actionsResult","execute","where","namespace","targetId","id","actions","value","items","action","cancelRes","isFail","CancelScheduledActionOnRevisionDeleteEventHandler","createImplementation","implementation","dependencies"],"sources":["CancelScheduledActionOnRevisionDeleteEventHandler.ts"],"sourcesContent":["import { EntryAfterDeleteEventHandler } from \"@webiny/api-headless-cms/features/contentEntry/DeleteEntry/events\";\nimport {\n CancelScheduledActionUseCase,\n ListScheduledActionsUseCase\n} from \"@webiny/api-scheduler/exports/api/scheduler.js\";\nimport { createNamespace } from \"~/utils/namespace.js\";\n\n/**\n * Cancels scheduled actions when an entry revision is deleted\n *\n * When a user deletes an entry revision, any scheduled publish/unpublish\n * action for that revision should be canceled since the revision\n * no longer exists.\n */\nclass CancelScheduledActionOnDeleteHandlerImpl implements EntryAfterDeleteEventHandler.Interface {\n constructor(\n private listScheduledActions: ListScheduledActionsUseCase.Interface,\n private cancelScheduledAction: CancelScheduledActionUseCase.Interface\n ) {}\n\n async handle(event: EntryAfterDeleteEventHandler.Event): Promise<void> {\n const { entry, model } = event.payload;\n\n // Skip private models\n if (model.isPrivate) {\n return;\n }\n\n const actionsResult = await this.listScheduledActions.execute({\n where: {\n namespace: createNamespace(model),\n targetId: entry.id\n }\n });\n\n const actions = actionsResult.value.items;\n\n for (const action of actions) {\n const cancelRes = await this.cancelScheduledAction.execute(action);\n if (cancelRes.isFail()) {\n // Silently ignore errors - this is non-critical cleanup.\n // Entry was deleted successfully, cancelling scheduled actions is best-effort.\n }\n }\n }\n}\n\nexport const CancelScheduledActionOnRevisionDeleteEventHandler =\n EntryAfterDeleteEventHandler.createImplementation({\n implementation: CancelScheduledActionOnDeleteHandlerImpl,\n dependencies: [ListScheduledActionsUseCase, CancelScheduledActionUseCase]\n });\n"],"mappings":"AAAA,SAASA,4BAA4B,QAAQ,mEAAmE;AAChH,SACIC,4BAA4B,EAC5BC,2BAA2B,QACxB,gDAAgD;AACvD,SAASC,eAAe;;AAExB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAMC,wCAAwC,CAAmD;EAC7FC,WAAWA,CACCC,oBAA2D,EAC3DC,qBAA6D,EACvE;IAAA,KAFUD,oBAA2D,GAA3DA,oBAA2D;IAAA,KAC3DC,qBAA6D,GAA7DA,qBAA6D;EACtE;EAEH,MAAMC,MAAMA,CAACC,KAAyC,EAAiB;IACnE,MAAM;MAAEC,KAAK;MAAEC;IAAM,CAAC,GAAGF,KAAK,CAACG,OAAO;;IAEtC;IACA,IAAID,KAAK,CAACE,SAAS,EAAE;MACjB;IACJ;IAEA,MAAMC,aAAa,GAAG,MAAM,IAAI,CAACR,oBAAoB,CAACS,OAAO,CAAC;MAC1DC,KAAK,EAAE;QACHC,SAAS,EAAEd,eAAe,CAACQ,KAAK,CAAC;QACjCO,QAAQ,EAAER,KAAK,CAACS;MACpB;IACJ,CAAC,CAAC;IAEF,MAAMC,OAAO,GAAGN,aAAa,CAACO,KAAK,CAACC,KAAK;IAEzC,KAAK,MAAMC,MAAM,IAAIH,OAAO,EAAE;MAC1B,MAAMI,SAAS,GAAG,MAAM,IAAI,CAACjB,qBAAqB,CAACQ,OAAO,CAACQ,MAAM,CAAC;MAClE,IAAIC,SAAS,CAACC,MAAM,CAAC,CAAC,EAAE;QACpB;QACA;MAAA;IAER;EACJ;AACJ;AAEA,OAAO,MAAMC,iDAAiD,GAC1D1B,4BAA4B,CAAC2B,oBAAoB,CAAC;EAC9CC,cAAc,EAAExB,wCAAwC;EACxDyB,YAAY,EAAE,CAAC3B,2BAA2B,EAAED,4BAA4B;AAC5E,CAAC,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { EntryAfterUnpublishEventHandler } from "@webiny/api-headless-cms/features/contentEntry/UnpublishEntry/events";
|
|
2
|
+
import { CancelScheduledActionUseCase, ListScheduledActionsUseCase } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
3
|
+
/**
|
|
4
|
+
* Cancels scheduled action when an entry is manually unpublished
|
|
5
|
+
*
|
|
6
|
+
* When a user manually unpublishes an entry revision, any scheduled unpublish
|
|
7
|
+
* action for that revision should be canceled since the manual action
|
|
8
|
+
* takes precedence.
|
|
9
|
+
*/
|
|
10
|
+
declare class CancelScheduledActionOnUnpublishHandlerImpl implements EntryAfterUnpublishEventHandler.Interface {
|
|
11
|
+
private listScheduledActions;
|
|
12
|
+
private cancelScheduledAction;
|
|
13
|
+
constructor(listScheduledActions: ListScheduledActionsUseCase.Interface, cancelScheduledAction: CancelScheduledActionUseCase.Interface);
|
|
14
|
+
handle(event: EntryAfterUnpublishEventHandler.Event): Promise<void>;
|
|
15
|
+
}
|
|
16
|
+
export declare const CancelScheduledActionOnUnpublishEventHandler: typeof CancelScheduledActionOnUnpublishHandlerImpl & {
|
|
17
|
+
__abstraction: import("@webiny/di").Abstraction<import("@webiny/api-core/features/eventPublisher").IEventHandler<import("@webiny/api-headless-cms/features/contentEntry/UnpublishEntry/events").EntryAfterUnpublishEvent>>;
|
|
18
|
+
};
|
|
19
|
+
export {};
|
package/features/CancelScheduledActionOnEntryChange/CancelScheduledActionOnUnpublishEventHandler.js
ADDED
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { EntryAfterUnpublishEventHandler } from "@webiny/api-headless-cms/features/contentEntry/UnpublishEntry/events";
|
|
2
|
+
import { CancelScheduledActionUseCase, ListScheduledActionsUseCase, ScheduledActionTypeUnpublish } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
3
|
+
import { createNamespace } from "../../utils/namespace.js";
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Cancels scheduled action when an entry is manually unpublished
|
|
7
|
+
*
|
|
8
|
+
* When a user manually unpublishes an entry revision, any scheduled unpublish
|
|
9
|
+
* action for that revision should be canceled since the manual action
|
|
10
|
+
* takes precedence.
|
|
11
|
+
*/
|
|
12
|
+
class CancelScheduledActionOnUnpublishHandlerImpl {
|
|
13
|
+
constructor(listScheduledActions, cancelScheduledAction) {
|
|
14
|
+
this.listScheduledActions = listScheduledActions;
|
|
15
|
+
this.cancelScheduledAction = cancelScheduledAction;
|
|
16
|
+
}
|
|
17
|
+
async handle(event) {
|
|
18
|
+
const {
|
|
19
|
+
entry,
|
|
20
|
+
model
|
|
21
|
+
} = event.payload;
|
|
22
|
+
|
|
23
|
+
// Skip private models
|
|
24
|
+
if (model.isPrivate) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
const actionsResult = await this.listScheduledActions.execute({
|
|
28
|
+
where: {
|
|
29
|
+
namespace: createNamespace(model),
|
|
30
|
+
actionType: ScheduledActionTypeUnpublish,
|
|
31
|
+
targetId: entry.id
|
|
32
|
+
}
|
|
33
|
+
});
|
|
34
|
+
const actions = actionsResult.value.items;
|
|
35
|
+
for (const action of actions) {
|
|
36
|
+
const cancelRes = await this.cancelScheduledAction.execute(action);
|
|
37
|
+
if (cancelRes.isFail()) {
|
|
38
|
+
// Silently ignore errors - this is non-critical cleanup.
|
|
39
|
+
// Entry was unpublished successfully, cancelling scheduled actions is best-effort.
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
export const CancelScheduledActionOnUnpublishEventHandler = EntryAfterUnpublishEventHandler.createImplementation({
|
|
45
|
+
implementation: CancelScheduledActionOnUnpublishHandlerImpl,
|
|
46
|
+
dependencies: [ListScheduledActionsUseCase, CancelScheduledActionUseCase]
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
//# sourceMappingURL=CancelScheduledActionOnUnpublishEventHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["EntryAfterUnpublishEventHandler","CancelScheduledActionUseCase","ListScheduledActionsUseCase","ScheduledActionTypeUnpublish","createNamespace","CancelScheduledActionOnUnpublishHandlerImpl","constructor","listScheduledActions","cancelScheduledAction","handle","event","entry","model","payload","isPrivate","actionsResult","execute","where","namespace","actionType","targetId","id","actions","value","items","action","cancelRes","isFail","CancelScheduledActionOnUnpublishEventHandler","createImplementation","implementation","dependencies"],"sources":["CancelScheduledActionOnUnpublishEventHandler.ts"],"sourcesContent":["import { EntryAfterUnpublishEventHandler } from \"@webiny/api-headless-cms/features/contentEntry/UnpublishEntry/events\";\nimport {\n CancelScheduledActionUseCase,\n ListScheduledActionsUseCase,\n ScheduledActionTypeUnpublish\n} from \"@webiny/api-scheduler/exports/api/scheduler.js\";\nimport { createNamespace } from \"~/utils/namespace.js\";\n\n/**\n * Cancels scheduled action when an entry is manually unpublished\n *\n * When a user manually unpublishes an entry revision, any scheduled unpublish\n * action for that revision should be canceled since the manual action\n * takes precedence.\n */\nclass CancelScheduledActionOnUnpublishHandlerImpl\n implements EntryAfterUnpublishEventHandler.Interface\n{\n constructor(\n private listScheduledActions: ListScheduledActionsUseCase.Interface,\n private cancelScheduledAction: CancelScheduledActionUseCase.Interface\n ) {}\n\n async handle(event: EntryAfterUnpublishEventHandler.Event): Promise<void> {\n const { entry, model } = event.payload;\n\n // Skip private models\n if (model.isPrivate) {\n return;\n }\n\n const actionsResult = await this.listScheduledActions.execute({\n where: {\n namespace: createNamespace(model),\n actionType: ScheduledActionTypeUnpublish,\n targetId: entry.id\n }\n });\n\n const actions = actionsResult.value.items;\n\n for (const action of actions) {\n const cancelRes = await this.cancelScheduledAction.execute(action);\n if (cancelRes.isFail()) {\n // Silently ignore errors - this is non-critical cleanup.\n // Entry was unpublished successfully, cancelling scheduled actions is best-effort.\n }\n }\n }\n}\n\nexport const CancelScheduledActionOnUnpublishEventHandler =\n EntryAfterUnpublishEventHandler.createImplementation({\n implementation: CancelScheduledActionOnUnpublishHandlerImpl,\n dependencies: [ListScheduledActionsUseCase, CancelScheduledActionUseCase]\n });\n"],"mappings":"AAAA,SAASA,+BAA+B,QAAQ,sEAAsE;AACtH,SACIC,4BAA4B,EAC5BC,2BAA2B,EAC3BC,4BAA4B,QACzB,gDAAgD;AACvD,SAASC,eAAe;;AAExB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAMC,2CAA2C,CAEjD;EACIC,WAAWA,CACCC,oBAA2D,EAC3DC,qBAA6D,EACvE;IAAA,KAFUD,oBAA2D,GAA3DA,oBAA2D;IAAA,KAC3DC,qBAA6D,GAA7DA,qBAA6D;EACtE;EAEH,MAAMC,MAAMA,CAACC,KAA4C,EAAiB;IACtE,MAAM;MAAEC,KAAK;MAAEC;IAAM,CAAC,GAAGF,KAAK,CAACG,OAAO;;IAEtC;IACA,IAAID,KAAK,CAACE,SAAS,EAAE;MACjB;IACJ;IAEA,MAAMC,aAAa,GAAG,MAAM,IAAI,CAACR,oBAAoB,CAACS,OAAO,CAAC;MAC1DC,KAAK,EAAE;QACHC,SAAS,EAAEd,eAAe,CAACQ,KAAK,CAAC;QACjCO,UAAU,EAAEhB,4BAA4B;QACxCiB,QAAQ,EAAET,KAAK,CAACU;MACpB;IACJ,CAAC,CAAC;IAEF,MAAMC,OAAO,GAAGP,aAAa,CAACQ,KAAK,CAACC,KAAK;IAEzC,KAAK,MAAMC,MAAM,IAAIH,OAAO,EAAE;MAC1B,MAAMI,SAAS,GAAG,MAAM,IAAI,CAAClB,qBAAqB,CAACQ,OAAO,CAACS,MAAM,CAAC;MAClE,IAAIC,SAAS,CAACC,MAAM,CAAC,CAAC,EAAE;QACpB;QACA;MAAA;IAER;EACJ;AACJ;AAEA,OAAO,MAAMC,4CAA4C,GACrD5B,+BAA+B,CAAC6B,oBAAoB,CAAC;EACjDC,cAAc,EAAEzB,2CAA2C;EAC3D0B,YAAY,EAAE,CAAC7B,2BAA2B,EAAED,4BAA4B;AAC5E,CAAC,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CancelScheduledActionOnEntryChange Feature
|
|
3
|
+
*
|
|
4
|
+
* Automatically cancels scheduled actions when entries are manually
|
|
5
|
+
* published, unpublished, or deleted. This ensures scheduled actions
|
|
6
|
+
* don't execute after a user has already performed the action manually.
|
|
7
|
+
*/
|
|
8
|
+
export declare const CancelScheduledActionOnEntryChangeFeature: {
|
|
9
|
+
name: string;
|
|
10
|
+
register(container: import("@webiny/di").Container): void;
|
|
11
|
+
};
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { createFeature } from "@webiny/feature/api";
|
|
2
|
+
import { CancelScheduledActionOnPublishEventHandler } from "./CancelScheduledActionOnPublishEventHandler.js";
|
|
3
|
+
import { CancelScheduledActionOnUnpublishEventHandler } from "./CancelScheduledActionOnUnpublishEventHandler.js";
|
|
4
|
+
import { CancelScheduledActionOnEntryDeleteEventHandler } from "./CancelScheduledActionOnEntryDeleteEventHandler.js";
|
|
5
|
+
import { CancelScheduledActionOnRevisionDeleteEventHandler } from "./CancelScheduledActionOnRevisionDeleteEventHandler.js";
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* CancelScheduledActionOnEntryChange Feature
|
|
9
|
+
*
|
|
10
|
+
* Automatically cancels scheduled actions when entries are manually
|
|
11
|
+
* published, unpublished, or deleted. This ensures scheduled actions
|
|
12
|
+
* don't execute after a user has already performed the action manually.
|
|
13
|
+
*/
|
|
14
|
+
export const CancelScheduledActionOnEntryChangeFeature = createFeature({
|
|
15
|
+
name: "CancelScheduledActionOnEntryChange",
|
|
16
|
+
register(container) {
|
|
17
|
+
container.register(CancelScheduledActionOnPublishEventHandler);
|
|
18
|
+
container.register(CancelScheduledActionOnUnpublishEventHandler);
|
|
19
|
+
container.register(CancelScheduledActionOnEntryDeleteEventHandler);
|
|
20
|
+
container.register(CancelScheduledActionOnRevisionDeleteEventHandler);
|
|
21
|
+
}
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
//# sourceMappingURL=feature.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["createFeature","CancelScheduledActionOnPublishEventHandler","CancelScheduledActionOnUnpublishEventHandler","CancelScheduledActionOnEntryDeleteEventHandler","CancelScheduledActionOnRevisionDeleteEventHandler","CancelScheduledActionOnEntryChangeFeature","name","register","container"],"sources":["feature.ts"],"sourcesContent":["import { createFeature } from \"@webiny/feature/api\";\nimport { CancelScheduledActionOnPublishEventHandler } from \"./CancelScheduledActionOnPublishEventHandler.js\";\nimport { CancelScheduledActionOnUnpublishEventHandler } from \"./CancelScheduledActionOnUnpublishEventHandler.js\";\nimport { CancelScheduledActionOnEntryDeleteEventHandler } from \"./CancelScheduledActionOnEntryDeleteEventHandler.js\";\nimport { CancelScheduledActionOnRevisionDeleteEventHandler } from \"./CancelScheduledActionOnRevisionDeleteEventHandler.js\";\n\n/**\n * CancelScheduledActionOnEntryChange Feature\n *\n * Automatically cancels scheduled actions when entries are manually\n * published, unpublished, or deleted. This ensures scheduled actions\n * don't execute after a user has already performed the action manually.\n */\nexport const CancelScheduledActionOnEntryChangeFeature = createFeature({\n name: \"CancelScheduledActionOnEntryChange\",\n register(container) {\n container.register(CancelScheduledActionOnPublishEventHandler);\n container.register(CancelScheduledActionOnUnpublishEventHandler);\n container.register(CancelScheduledActionOnEntryDeleteEventHandler);\n container.register(CancelScheduledActionOnRevisionDeleteEventHandler);\n }\n});\n"],"mappings":"AAAA,SAASA,aAAa,QAAQ,qBAAqB;AACnD,SAASC,0CAA0C;AACnD,SAASC,4CAA4C;AACrD,SAASC,8CAA8C;AACvD,SAASC,iDAAiD;;AAE1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO,MAAMC,yCAAyC,GAAGL,aAAa,CAAC;EACnEM,IAAI,EAAE,oCAAoC;EAC1CC,QAAQA,CAACC,SAAS,EAAE;IAChBA,SAAS,CAACD,QAAQ,CAACN,0CAA0C,CAAC;IAC9DO,SAAS,CAACD,QAAQ,CAACL,4CAA4C,CAAC;IAChEM,SAAS,CAACD,QAAQ,CAACJ,8CAA8C,CAAC;IAClEK,SAAS,CAACD,QAAQ,CAACH,iDAAiD,CAAC;EACzE;AACJ,CAAC,CAAC","ignoreList":[]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { NamespaceHandler as NamespaceHandlerAbstraction } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
2
|
+
import { GetModelUseCase } from "@webiny/api-headless-cms/features/contentModel/GetModel/index.js";
|
|
3
|
+
import { GetEntryByIdUseCase } from "@webiny/api-headless-cms/features/contentEntry/GetEntryById/index.js";
|
|
4
|
+
import type { GenericRecord } from "@webiny/api/types.js";
|
|
5
|
+
declare class NamespaceHandlerImpl implements NamespaceHandlerAbstraction.Interface<GenericRecord> {
|
|
6
|
+
private getModelUseCase;
|
|
7
|
+
private getEntryByIdUseCase;
|
|
8
|
+
constructor(getModelUseCase: GetModelUseCase.Interface, getEntryByIdUseCase: GetEntryByIdUseCase.Interface);
|
|
9
|
+
canHandle(namespace: string): boolean;
|
|
10
|
+
execute(params: NamespaceHandlerAbstraction.Params): NamespaceHandlerAbstraction.Response<GenericRecord>;
|
|
11
|
+
}
|
|
12
|
+
export declare const NamespaceHandler: typeof NamespaceHandlerImpl & {
|
|
13
|
+
__abstraction: import("@webiny/di").Abstraction<import("@webiny/api-scheduler/features/NamespaceHandler/abstractions").INamespaceHandler<GenericRecord>>;
|
|
14
|
+
};
|
|
15
|
+
export {};
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { NamespaceHandler as NamespaceHandlerAbstraction } from "@webiny/api-scheduler/exports/api/scheduler.js";
|
|
2
|
+
import { extractModelIdFromNamespace } from "../../utils/namespace.js";
|
|
3
|
+
import { Result } from "@webiny/feature/exports/api.js";
|
|
4
|
+
import { GetModelUseCase } from "@webiny/api-headless-cms/features/contentModel/GetModel/index.js";
|
|
5
|
+
import { GetEntryByIdUseCase } from "@webiny/api-headless-cms/features/contentEntry/GetEntryById/index.js";
|
|
6
|
+
class NamespaceHandlerImpl {
|
|
7
|
+
constructor(getModelUseCase, getEntryByIdUseCase) {
|
|
8
|
+
this.getModelUseCase = getModelUseCase;
|
|
9
|
+
this.getEntryByIdUseCase = getEntryByIdUseCase;
|
|
10
|
+
}
|
|
11
|
+
canHandle(namespace) {
|
|
12
|
+
const modelId = extractModelIdFromNamespace(namespace);
|
|
13
|
+
return !!modelId;
|
|
14
|
+
}
|
|
15
|
+
async execute(params) {
|
|
16
|
+
/**
|
|
17
|
+
* We know that modelId is ok because of the canHandle() method, which is always called before execute() and must return true for this handler to be executed.
|
|
18
|
+
*/
|
|
19
|
+
const modelId = extractModelIdFromNamespace(params.namespace);
|
|
20
|
+
// Fetch the target model
|
|
21
|
+
const modelResult = await this.getModelUseCase.execute(modelId);
|
|
22
|
+
if (modelResult.isFail()) {
|
|
23
|
+
return Result.fail(modelResult.error);
|
|
24
|
+
}
|
|
25
|
+
const model = modelResult.value;
|
|
26
|
+
|
|
27
|
+
// Fetch entry to get title
|
|
28
|
+
const entryResult = await this.getEntryByIdUseCase.execute(model, params.targetId);
|
|
29
|
+
if (entryResult.isFail()) {
|
|
30
|
+
return Result.fail(entryResult.error);
|
|
31
|
+
}
|
|
32
|
+
const entry = entryResult.value;
|
|
33
|
+
const title = entry.values[model.titleFieldId] || "Unknown entry title";
|
|
34
|
+
return Result.ok({
|
|
35
|
+
namespace: params.namespace,
|
|
36
|
+
title,
|
|
37
|
+
modelId,
|
|
38
|
+
actionType: params.actionType,
|
|
39
|
+
targetId: params.targetId,
|
|
40
|
+
scheduleId: params.scheduleId,
|
|
41
|
+
immediately: params.immediately || false
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
export const NamespaceHandler = NamespaceHandlerAbstraction.createImplementation({
|
|
46
|
+
implementation: NamespaceHandlerImpl,
|
|
47
|
+
dependencies: [GetModelUseCase, GetEntryByIdUseCase]
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
//# sourceMappingURL=NamespaceHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"names":["NamespaceHandler","NamespaceHandlerAbstraction","extractModelIdFromNamespace","Result","GetModelUseCase","GetEntryByIdUseCase","NamespaceHandlerImpl","constructor","getModelUseCase","getEntryByIdUseCase","canHandle","namespace","modelId","execute","params","modelResult","isFail","fail","error","model","value","entryResult","targetId","entry","title","values","titleFieldId","ok","actionType","scheduleId","immediately","createImplementation","implementation","dependencies"],"sources":["NamespaceHandler.ts"],"sourcesContent":["import { NamespaceHandler as NamespaceHandlerAbstraction } from \"@webiny/api-scheduler/exports/api/scheduler.js\";\nimport { extractModelIdFromNamespace } from \"~/utils/namespace.js\";\nimport { Result } from \"@webiny/feature/exports/api.js\";\nimport { GetModelUseCase } from \"@webiny/api-headless-cms/features/contentModel/GetModel/index.js\";\nimport { GetEntryByIdUseCase } from \"@webiny/api-headless-cms/features/contentEntry/GetEntryById/index.js\";\nimport type { GenericRecord } from \"@webiny/api/types.js\";\n\nclass NamespaceHandlerImpl implements NamespaceHandlerAbstraction.Interface<GenericRecord> {\n public constructor(\n private getModelUseCase: GetModelUseCase.Interface,\n private getEntryByIdUseCase: GetEntryByIdUseCase.Interface\n ) {}\n\n public canHandle(namespace: string): boolean {\n const modelId = extractModelIdFromNamespace(namespace);\n\n return !!modelId;\n }\n\n public async execute(\n params: NamespaceHandlerAbstraction.Params\n ): NamespaceHandlerAbstraction.Response<GenericRecord> {\n /**\n * We know that modelId is ok because of the canHandle() method, which is always called before execute() and must return true for this handler to be executed.\n */\n const modelId = extractModelIdFromNamespace(params.namespace)!;\n // Fetch the target model\n const modelResult = await this.getModelUseCase.execute(modelId);\n if (modelResult.isFail()) {\n return Result.fail(modelResult.error as any);\n }\n\n const model = modelResult.value;\n\n // Fetch entry to get title\n const entryResult = await this.getEntryByIdUseCase.execute(model, params.targetId);\n if (entryResult.isFail()) {\n return Result.fail(entryResult.error as any);\n }\n\n const entry = entryResult.value;\n const title = entry.values[model.titleFieldId] || \"Unknown entry title\";\n\n return Result.ok({\n namespace: params.namespace,\n title,\n modelId,\n actionType: params.actionType,\n targetId: params.targetId,\n scheduleId: params.scheduleId,\n immediately: params.immediately || false\n });\n }\n}\n\nexport const NamespaceHandler = NamespaceHandlerAbstraction.createImplementation({\n implementation: NamespaceHandlerImpl,\n dependencies: [GetModelUseCase, GetEntryByIdUseCase]\n});\n"],"mappings":"AAAA,SAASA,gBAAgB,IAAIC,2BAA2B,QAAQ,gDAAgD;AAChH,SAASC,2BAA2B;AACpC,SAASC,MAAM,QAAQ,gCAAgC;AACvD,SAASC,eAAe,QAAQ,kEAAkE;AAClG,SAASC,mBAAmB,QAAQ,sEAAsE;AAG1G,MAAMC,oBAAoB,CAAiE;EAChFC,WAAWA,CACNC,eAA0C,EAC1CC,mBAAkD,EAC5D;IAAA,KAFUD,eAA0C,GAA1CA,eAA0C;IAAA,KAC1CC,mBAAkD,GAAlDA,mBAAkD;EAC3D;EAEIC,SAASA,CAACC,SAAiB,EAAW;IACzC,MAAMC,OAAO,GAAGV,2BAA2B,CAACS,SAAS,CAAC;IAEtD,OAAO,CAAC,CAACC,OAAO;EACpB;EAEA,MAAaC,OAAOA,CAChBC,MAA0C,EACS;IACnD;AACR;AACA;IACQ,MAAMF,OAAO,GAAGV,2BAA2B,CAACY,MAAM,CAACH,SAAS,CAAE;IAC9D;IACA,MAAMI,WAAW,GAAG,MAAM,IAAI,CAACP,eAAe,CAACK,OAAO,CAACD,OAAO,CAAC;IAC/D,IAAIG,WAAW,CAACC,MAAM,CAAC,CAAC,EAAE;MACtB,OAAOb,MAAM,CAACc,IAAI,CAACF,WAAW,CAACG,KAAY,CAAC;IAChD;IAEA,MAAMC,KAAK,GAAGJ,WAAW,CAACK,KAAK;;IAE/B;IACA,MAAMC,WAAW,GAAG,MAAM,IAAI,CAACZ,mBAAmB,CAACI,OAAO,CAACM,KAAK,EAAEL,MAAM,CAACQ,QAAQ,CAAC;IAClF,IAAID,WAAW,CAACL,MAAM,CAAC,CAAC,EAAE;MACtB,OAAOb,MAAM,CAACc,IAAI,CAACI,WAAW,CAACH,KAAY,CAAC;IAChD;IAEA,MAAMK,KAAK,GAAGF,WAAW,CAACD,KAAK;IAC/B,MAAMI,KAAK,GAAGD,KAAK,CAACE,MAAM,CAACN,KAAK,CAACO,YAAY,CAAC,IAAI,qBAAqB;IAEvE,OAAOvB,MAAM,CAACwB,EAAE,CAAC;MACbhB,SAAS,EAAEG,MAAM,CAACH,SAAS;MAC3Ba,KAAK;MACLZ,OAAO;MACPgB,UAAU,EAAEd,MAAM,CAACc,UAAU;MAC7BN,QAAQ,EAAER,MAAM,CAACQ,QAAQ;MACzBO,UAAU,EAAEf,MAAM,CAACe,UAAU;MAC7BC,WAAW,EAAEhB,MAAM,CAACgB,WAAW,IAAI;IACvC,CAAC,CAAC;EACN;AACJ;AAEA,OAAO,MAAM9B,gBAAgB,GAAGC,2BAA2B,CAAC8B,oBAAoB,CAAC;EAC7EC,cAAc,EAAE1B,oBAAoB;EACpC2B,YAAY,EAAE,CAAC7B,eAAe,EAAEC,mBAAmB;AACvD,CAAC,CAAC","ignoreList":[]}
|