0.1.1
Prevalence for Deno, like Prevayler.
Attributes
Includes Deno configuration
Repository
Current version released
2 years ago
Versions
prevalence
System prevalence as a typed library, using Deno.Kv for storage.
TypeScript implementation for Deno of the Prevalence design pattern, as introduced by Klaus Wuestefeld in 1998 with Prevayler.
Saves periodical snapshots of the whole model for faster startup, and keeps a
journal of executed actions since last snapshot, using a
Persister. The Persister
uses a
Marshaller to serialize/deserialize the model
and the journal.
Requirements
Requires Deno v1.32 or later, with the --unstable
flag.
API
Please see the auto-generated API documentation.
Example usage
import {
Action,
KvPersister,
logger,
Marshaller,
Model,
Persister,
Prevalence,
SerializableClassesContainer,
SuperserialMarshaller,
} from "https://deno.land/x/prevalence/mod.ts";
import { Serializer } from "https://deno.land/x/superserial@0.3.4/serializer.ts";
const log = logger(import.meta.url);
class User {
constructor(
readonly uuid: string,
public displayName: string,
) {}
}
const alice: User = new User("1", "Alice");
type Post = {
id: string;
subject: string;
};
class MyModel implements Model<MyModel> {
constructor(
public posts: Record<string, Post>,
public users: Record<string, User>,
) {}
}
class AddPostAction implements Action<MyModel> {
constructor(public post: Post) {}
execute(model: MyModel): void {
model.posts[this.post.id] = this.post;
}
}
class RemovePostAction implements Action<MyModel> {
constructor(public postId: string) {}
execute(model: MyModel): void {
delete model.posts[this.postId];
}
}
class AddUserAction implements Action<MyModel> {
constructor(public user: User) {}
execute(model: MyModel): void {
model.users[this.user.uuid] = this.user;
}
}
class RemoveUserAction implements Action<MyModel> {
constructor(public userId: string) {}
execute(model: MyModel): void {
delete model.users[this.userId];
}
}
const classes: SerializableClassesContainer = {
User,
MyModel,
AddPostAction,
RemovePostAction,
AddUserAction,
RemoveUserAction,
};
const marshaller: Marshaller<MyModel, string> = new SuperserialMarshaller<
MyModel
>(
new Serializer({ classes }),
);
const kv: Deno.Kv = await Deno.openKv("example-person-invoice.db");
const persister: Persister<MyModel> = new KvPersister<MyModel, string>(
marshaller,
kv,
);
const defaultInitialModel: MyModel = { posts: {}, users: {} };
const prevalence: Prevalence<MyModel> = await Prevalence.create<MyModel>(
defaultInitialModel,
{ persister, classes },
);
await prevalence.execute(new AddPostAction({ id: "post#1", subject: "Lorem" }));
await prevalence.execute(new AddPostAction({ id: "post#2", subject: "Ipsum" }));
await prevalence.execute(new AddPostAction({ id: "post#3", subject: "Dolor" }));
await prevalence.execute(new RemovePostAction("post#2"));
await prevalence.execute(new AddUserAction(alice));
const posts: Post[] = Object.values(prevalence.model.posts);
log("Posts:");
for (const post of posts) {
log(`${post.id}: ${post.subject}`);
}
log("Users:");
for (const user of Object.values(prevalence.model.users)) {
log(`${user.uuid}: ${user.displayName}`);
}
await prevalence.execute(new RemoveUserAction(alice.uuid));
log("Users:");
for (const user of Object.values(prevalence.model.users)) {
log(`${user.uuid}: ${user.displayName}`);
}
await prevalence.snapshot();
log("Done.");
You may run the above example with:
deno run --unstable --reload --allow-write=example-person-invoice.db --allow-read=example-person-invoice.db https://deno.land/x/prevalence/readme/person-invoice.ts
For further usage examples, see the tests: