Compare commits

..

91 Commits

Author SHA1 Message Date
Jacob Nguyen
d1680e51fc change task signature 2024-07-18 16:47:13 -05:00
jacob
c35337c3cf fix swap not accepting functions 2024-07-16 23:09:24 -05:00
Jacob Nguyen
5d0260ab4e documentation 2024-07-07 18:21:16 -05:00
Jacob Nguyen
2014e0ea8e refactor n better signatures for task scheduler 2024-07-06 14:12:03 -05:00
Jacob Nguyen
482105712b refactor to not type leak and job cancellation 2024-07-05 17:18:26 -05:00
Jacob Nguyen
870972000d more descriptive errors 2024-07-05 15:48:55 -05:00
Jacob Nguyen
210aa41c7e scheduler ids 2024-07-05 14:46:31 -05:00
Jacob Nguyen
c252854954 revising cron modules and better error messages 2024-07-04 19:18:52 -05:00
Jacob Nguyen
92ca9eb859 namespace presence types again 2024-07-03 22:25:24 -05:00
Jacob Nguyen
e0f631a8e2 temp fix 2024-06-30 17:39:00 -05:00
Jacob Nguyen
3128b441c5 better documentation 2024-06-30 17:39:00 -05:00
Jacob Nguyen
14e80016da Update src/core/functions.ts
Co-authored-by: Evo <85353424+EvolutionX-10@users.noreply.github.com>
2024-06-30 17:32:08 -05:00
jacob
908d584cc5 fix 2024-06-24 23:18:19 -05:00
Jacob Nguyen
3e9b9229c8 Merge branch 'main' into feat/v4 2024-06-22 14:20:29 -05:00
Jacob Nguyen
1d8dbb8962 fix up tests and cleanup 2024-06-22 12:10:52 -05:00
Jacob Nguyen
7c8e39defb fix tests 2024-06-14 20:31:05 -05:00
Jacob Nguyen
90f5ea7bda update locals api, docs, tests 2024-06-14 20:27:23 -05:00
Jacob Nguyen
2f778f4dc2 s 2024-06-13 13:28:15 -05:00
Jacob Nguyen
9c358e1928 nvm, now i did 2024-06-13 13:28:03 -05:00
Jacob Nguyen
c764de12ac finish ioc transition 2024-06-13 12:26:37 -05:00
Jacob Nguyen
19abb7cb22 ss 2024-06-12 23:09:28 -05:00
Jacob Nguyen
30feb790b1 remove asset 2024-06-12 23:09:13 -05:00
Jacob Nguyen
222ecd9b61 documentation, add json for Asset 2024-06-11 00:27:36 -05:00
Jacob Nguyen
67bb4d4b9f fix: ioc deps not created correctly 2024-06-10 00:22:10 -05:00
Jacob Nguyen
bf071b7af4 -file,-updateModule,publish? 2024-06-08 00:20:24 -05:00
Jacob Nguyen
45665292ae add init hooks not firing 2024-06-04 00:48:30 -05:00
Jacob Nguyen
2120b18c4e more intuitive context.options and Asset typings 2024-06-02 23:43:55 -05:00
Jacob Nguyen
898fdf52a3 assets fn complete 2024-05-31 00:48:51 -05:00
Jacob Nguyen
c8230334f2 fix typings 2024-05-28 13:43:56 -05:00
Jacob Nguyen
792015a64e readd vitest and Asset fn 2024-05-27 17:01:22 -05:00
jacob
fb418c0675 refact,simplf 2024-05-25 16:22:25 -05:00
jacob
12a8f0c5d7 once on eventModules 2024-05-25 12:39:24 -05:00
Jacob Nguyen
af0f909c44 consolidate fmt 2024-05-23 22:00:12 -05:00
jacob
76d9db7b98 inlineinignsd 2024-05-23 19:36:08 -05:00
Jacob Nguyen
86dd0cd842 testing n shi 2024-05-23 01:02:02 -05:00
Jacob Nguyen
1de21b8b37 handling customId params working 2024-05-22 23:33:42 -05:00
Jacob Nguyen
327e56fc1c dynamic customIds 2024-05-22 00:44:59 -05:00
Jacob Nguyen
814fc4f01d proper mapping 2024-05-21 01:41:34 -05:00
Jacob Nguyen
7f4004e043 parsingParams kinda 2024-05-21 01:18:20 -05:00
Jacob Nguyen
88598b0948 fix initPlugins not reassigning 2024-05-21 00:35:19 -05:00
Jacob Nguyen
e700297bfc fix faiklling test 2024-05-20 19:36:36 -05:00
Jacob Nguyen
e0f6a4cd16 initplugins inject deps, inconspicuos 2024-05-20 12:21:18 -05:00
Jacob Nguyen
15511a4868 porg 2024-05-20 00:49:51 -05:00
Jacob Nguyen
735a9e3816 plugin system loking better, tbd type 2024-05-19 22:50:21 -05:00
Jacob Nguyen
0beeb4c064 add deps to plugin calls and execute 2024-05-19 22:33:57 -05:00
Jacob Nguyen
699adf276c simplify plugin args and prepare for reduction among plugins 2024-05-17 23:13:12 -05:00
Jacob Nguyen
960f90c544 freeze module after plugins, updateModule, and more 2024-05-17 20:16:16 -05:00
Jacob Nguyen
6717672722 plugin data reduction & args changes 2024-05-17 15:07:32 -05:00
Jacob Nguyen
ca9b84ba21 save b4 thunder and lightning 2024-05-16 23:41:44 -05:00
Jacob Nguyen
d3227e5ec1 updating ioc api, experimenting with cron 2024-05-16 17:03:04 -05:00
Jacob Nguyen
44c072f401 ksdjkldsfld 2024-05-16 00:21:29 -05:00
Jacob Nguyen
203e8c8ecf cron works now, poc 2024-05-15 16:52:33 -05:00
Jacob Nguyen
d905f08993 i think cron works 2024-05-15 16:42:21 -05:00
Jacob Nguyen
0d82658fc5 up to speed with event modules 2024-05-15 14:59:24 -05:00
Jacob Nguyen
16a84e85d1 add more tests, polish up ioc 2024-05-15 01:44:23 -05:00
Jacob Nguyen
ec45f80be6 refactor deps list 2024-05-15 00:29:59 -05:00
Jacob Nguyen
eb8ba6799b fdssdf 2024-05-14 23:29:20 -05:00
Jacob Nguyen
a7aea4be1a ready handler revamped so much cleaner 2024-05-14 23:19:57 -05:00
Jacob Nguyen
880311f08c refactor, add cron types, reinstante module loader 2024-05-14 12:01:18 -05:00
Jacob Nguyen
0a05cbba3f dsfsd 2024-05-13 16:06:02 -05:00
Jacob Nguyen
6e2f4b616f refactor and clean up and reenter v3 module loading 2024-05-13 15:27:12 -05:00
Jacob Nguyen
8554eeaef4 working on bun compat 2024-05-10 01:18:44 -05:00
Jacob Nguyen
4d74e63dd3 fix absPath gen 2024-05-10 00:23:22 -05:00
jacob
ffb2872f8b ensure container is init'd 2024-05-09 20:42:09 -05:00
Jacob Nguyen
08ef80522f fdsfD 2024-05-04 12:24:22 -05:00
Jacob Nguyen
f762033504 removeiti 2024-05-03 18:50:47 -05:00
Jacob Nguyen
cd92b54839 for now copy paste new ioc system 2024-05-03 18:13:34 -05:00
Jacob Nguyen
b0e9d15fa7 generify partition 2024-05-03 17:35:25 -05:00
jacob
545105e45b some refactor 2024-05-03 16:14:55 -05:00
jacob
d7ebdb2edc cleanup tests, codegen, and importing handler 2024-05-02 17:27:57 -05:00
Jacob Nguyen
d3245c8a0c prototyping linking static handler 2024-05-02 01:22:40 -05:00
jacob
219eda9bf7 more refactoring 2024-05-01 18:24:56 -05:00
jacob
0488f45677 refactor allat 2024-05-01 16:46:12 -05:00
jacob
e8cfcc2525 reffactor 2024-04-30 20:31:06 -05:00
jacob
d6eedb1b4d remove barrel for core/structs 2024-04-30 20:23:17 -05:00
jacob
5b33a9d1bb still broken but progress 2024-04-30 15:03:19 -05:00
Jacob Nguyen
52d1b5a37a fix circular dependency 2024-04-29 00:48:37 -05:00
Jacob Nguyen
6c61ae7d6a fix test 2024-04-29 00:17:58 -05:00
Jacob Nguyen
87c5631e57 clean up dependencies 2024-04-29 00:13:16 -05:00
Jacob Nguyen
821de65b86 internal namespace 2024-04-29 00:08:57 -05:00
Jacob Nguyen
e6fba9d8b5 Presence namespaced types removed 2024-04-28 23:37:58 -05:00
Jacob Nguyen
071d5eac49 the end of sern init?? 2024-04-28 22:16:54 -05:00
Jacob Nguyen
68c5f09b46 fix text compile 2024-04-28 21:20:36 -05:00
Jacob Nguyen
6105f7b4ef TEAR IT UP 2024-04-28 20:50:49 -05:00
Jacob Nguyen
9faae7eca7 consolidate default services in single file 2024-04-28 20:36:50 -05:00
Jacob Nguyen
f5136ba1ca consolidate interfaces in single file 2024-04-28 20:26:53 -05:00
Jacob Nguyen
76ee9c6edf Remove module store, manager, and Intializable type 2024-04-28 19:34:08 -05:00
Jacob Nguyen
3d10ee1c59 tearing it up 2024-04-28 18:44:58 -05:00
Jacob Nguyen
599a02c9df command modules do not depend on anything but itself 2024-04-28 15:50:29 -05:00
Jacob Nguyen
6228f53244 Refactorings 2024-04-28 14:14:51 -05:00
jacob
30230d49c3 step 1 2024-04-28 13:48:50 -05:00
17 changed files with 124 additions and 134 deletions

View File

@@ -1,39 +1,5 @@
# Changelog
## [4.0.3](https://github.com/sern-handler/handler/compare/v4.0.2...v4.0.3) (2024-10-06)
### Bug Fixes
* async presence ([#369](https://github.com/sern-handler/handler/issues/369)) ([eabfb81](https://github.com/sern-handler/handler/commit/eabfb81819b53a4656d8eac6e21cfb488b724a42))
* fix eventModule typing for Discord events ([#368](https://github.com/sern-handler/handler/issues/368)) ([1789ccb](https://github.com/sern-handler/handler/commit/1789ccb2f22f502f87538fecdb07106ff7110434))
## [4.0.2](https://github.com/sern-handler/handler/compare/v4.0.1...v4.0.2) (2024-08-13)
### Bug Fixes
* type issue ([2106cdc](https://github.com/sern-handler/handler/commit/2106cdc1d033f88b6ee4ccca6754fe7a595a9328))
## [4.0.1](https://github.com/sern-handler/handler/compare/v4.0.0...v4.0.1) (2024-07-19)
### Bug Fixes
* add SDT typings to autocomplete commands ([#363](https://github.com/sern-handler/handler/issues/363)) ([92623d2](https://github.com/sern-handler/handler/commit/92623d2914fb80e31365f06cf896bb37f36fc814))
## [4.0.0](https://github.com/sern-handler/handler/compare/v3.3.4...v4.0.0) (2024-07-18)
### Features
* v4 ([#361](https://github.com/sern-handler/handler/issues/361)) ([9a8904f](https://github.com/sern-handler/handler/commit/9a8904f5aed4fa36b018ad73bbe58049bae33274))
### Miscellaneous Chores
* release 4.0.0 ([dda0e33](https://github.com/sern-handler/handler/commit/dda0e3395b6704862bfd3fda2a201e2cb9b45d2f))
## [3.3.4](https://github.com/sern-handler/handler/compare/v3.3.3...v3.3.4) (2024-03-18)

View File

@@ -1,6 +1,6 @@
MIT License
Copyright (c) 2024 sern
Copyright (c) 2023 sern
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal

View File

@@ -25,7 +25,7 @@
- Unleash its full potential with a powerful CLI and awesome plugins.
## 📜 Installation
[Start here!!](https://sern.dev/v4/reference/getting-started)
[Start here!!](https://sern.dev/docs/guide/walkthrough/new-project)
## 👶 Basic Usage
<details><summary>ping.ts</summary>
@@ -48,6 +48,7 @@ export default commandModule({
- [Community Bot](https://github.com/sern-handler/sern-community), the community bot for our [discord server](https://sern.dev/discord).
- [Vinci](https://github.com/SrIzan10/vinci), the bot for Mara Turing.
- [Bask](https://github.com/baskbotml/bask), Listen your favorite artists on Discord.
- [ava](https://github.com/SrIzan10/ava), A discord bot that plays KNGI and Gensokyo Radio.
- [Murayama](https://github.com/murayamabot/murayama), :pepega:
- [Protector (WIP)](https://github.com/needhamgary/Protector), Just a simple bot to help enhance a private minecraft server.
- [SmokinWeed 💨](https://github.com/Peter-MJ-Parker/sern-bud), A fun bot for a small - but growing - server.

View File

@@ -1,7 +1,7 @@
{
"name": "@sern/handler",
"packageManager": "yarn@3.5.0",
"version": "4.0.3",
"version": "4.0.0",
"description": "A complete, customizable, typesafe, & reactive framework for discord bots.",
"main": "./dist/index.js",
"module": "./dist/index.js",
@@ -39,7 +39,8 @@
"callsites": "^3.1.0",
"cron": "^3.1.7",
"deepmerge": "^4.3.1",
"rxjs": "^7.8.0"
"rxjs": "^7.8.0",
"ts-results-es": "^4.1.0"
},
"devDependencies": {
"@faker-js/faker": "^8.0.1",

View File

@@ -26,7 +26,7 @@ export function commandModule(mod: InputCommand): Module {
* The wrapper function to define event modules for sern
* @param mod
*/
export function eventModule<T extends keyof ClientEvents = keyof ClientEvents>(mod: InputEvent<T>): Module {
export function eventModule(mod: InputEvent): Module {
const [onEvent, plugins] = partitionPlugins(mod.plugins);
if(onEvent.length !== 0) throw Error("Event modules cannot have ControlPlugins");
return { ...mod,
@@ -35,9 +35,8 @@ export function eventModule<T extends keyof ClientEvents = keyof ClientEvents>(m
}
/** Create event modules from discord.js client events,
* This was an {@link eventModule} for discord events,
* where typings were bad.
* @deprecated Use {@link eventModule} instead
* This is an {@link eventModule} for discord events,
* where typings can be very bad.
* @param mod
*/
export function discordEvent<T extends keyof ClientEvents>(mod: {

View File

@@ -1,6 +1,6 @@
import { CommandType, PluginType } from './structures/enums';
import type { Plugin, PluginResult, CommandArgs, InitArgs } from '../types/core-plugin';
import { Err, Ok } from './structures/result';
import { Err, Ok } from 'ts-results-es';
export function makePlugin<V extends unknown[]>(
type: PluginType,

View File

@@ -1,10 +1,11 @@
import type { ActivitiesOptions } from "discord.js";
import type { IntoDependencies } from "./ioc";
import type { Emitter } from "./interfaces";
import { Awaitable } from "../types/utility";
type Status = 'online' | 'idle' | 'invisible' | 'dnd'
type PresenceReduce = (previous: Presence.Result) => Awaitable<Presence.Result>;
type PresenceReduce = (previous: Presence.Result) => Presence.Result;
export const Presence = {
/**
@@ -49,7 +50,7 @@ export const Presence = {
export declare namespace Presence {
export type Config<T extends (keyof Dependencies)[]> = {
inject?: [...T]
execute: (...v: IntoDependencies<T>) => Awaitable<Presence.Result>;
execute: (...v: IntoDependencies<T>) => Presence.Result;
}
@@ -59,7 +60,7 @@ export declare namespace Presence {
activities?: ActivitiesOptions[];
shardId?: number[];
repeat?: number | [Emitter, string];
onRepeat?: PresenceReduce
onRepeat?: (previous: Result) => Result;
}
}

View File

@@ -8,7 +8,8 @@ import type {
Snowflake,
User,
} from 'discord.js';
import { Result, Ok, Err, val } from './result';
import { CoreContext } from '../structures/core-context';
import { Result, Ok, Err } from 'ts-results-es';
import * as assert from 'assert';
import type { ReplyOptions } from '../../types/utility';
import { fmt } from '../functions'
@@ -20,32 +21,39 @@ import { SernError } from './enums';
* Provides values shared between
* Message and ChatInputCommandInteraction
*/
export class Context {
export class Context extends CoreContext<Message, ChatInputCommandInteraction> {
get options() {
if(this.isMessage()) {
const [, ...rest] = fmt(this.message.content, this.prefix);
return rest;
}
return this.interaction.options;
} else {
return this.interaction.options;
}
}
protected constructor(protected ctx: Result<Message, ChatInputCommandInteraction>,
private __prefix?: string) { }
private __prefix?: string) {
super(ctx);
}
public get prefix() {
return this.__prefix;
}
public get id(): Snowflake {
return val(this.ctx).id
return safeUnwrap(this.ctx
.map(m => m.id)
.mapErr(i => i.id));
}
public get channel() {
return val(this.ctx).channel;
return safeUnwrap(this.ctx.map(m => m.channel).mapErr(i => i.channel));
}
public get channelId(): Snowflake {
return val(this.ctx).channelId;
return safeUnwrap(this.ctx
.map(m => m.channelId)
.mapErr(i => i.channelId));
}
/**
@@ -53,11 +61,9 @@ export class Context {
* else, interaction.user
*/
public get user(): User {
if(this.ctx.ok) {
return this.ctx.value.author;
}
return this.ctx.error.user;
return safeUnwrap(this.ctx
.map(m => m.author)
.mapErr(i => i.user));
}
public get userId(): Snowflake {
@@ -65,60 +71,59 @@ export class Context {
}
public get createdTimestamp(): number {
return val(this.ctx).createdTimestamp;
return safeUnwrap(this.ctx
.map(m => m.createdTimestamp)
.mapErr(i => i.createdTimestamp));
}
public get guild() {
return val(this.ctx).guild;
return safeUnwrap(this.ctx
.map(m => m.guild)
.mapErr(i => i.guild));
}
public get guildId() {
return val(this.ctx).guildId;
return safeUnwrap(this.ctx
.map(m => m.guildId)
.mapErr(i => i.guildId));
}
/*
* interactions can return APIGuildMember if the guild it is emitted from is not cached
*/
public get member() {
return val(this.ctx).member;
return safeUnwrap(this.ctx
.map(m => m.member)
.mapErr(i => i.member));
}
get message(): Message {
if(this.ctx.ok) {
return this.ctx.value;
}
throw Error(SernError.MismatchEvent);
}
public isMessage(): this is Context & { ctx: Result<Message, never> } {
return this.ctx.ok;
}
public isSlash(): this is Context & { ctx: Result<never, ChatInputCommandInteraction> } {
return !this.isMessage();
return this.ctx.expect(SernError.MismatchEvent);
}
get interaction(): ChatInputCommandInteraction {
if(!this.ctx.ok) {
return this.ctx.error;
}
throw Error(SernError.MismatchEvent);
return this.ctx.expectErr(SernError.MismatchEvent);
}
public get client(): Client {
return val(this.ctx).client;
return safeUnwrap(this.ctx
.map(m => m.client)
.mapErr(i => i.client));
}
public get inGuild(): boolean {
return val(this.ctx).inGuild()
return safeUnwrap(this.ctx
.map(m => m.inGuild())
.mapErr(i => i.inGuild()));
}
public async reply(content: ReplyOptions) {
if(this.ctx.ok) {
return this.ctx.value.reply(content as MessageReplyOptions)
}
interface FetchReply { fetchReply: true };
return this.ctx.error.reply(content as InteractionReplyOptions & FetchReply)
return safeUnwrap(
this.ctx
.map(m => m.reply(content as MessageReplyOptions))
.mapErr(i =>
i.reply(content as InteractionReplyOptions).then(() => i.fetchReply())),
);
}
static wrap(wrappable: BaseInteraction | Message, prefix?: string): Context {
@@ -129,3 +134,10 @@ export class Context {
return new Context(Err(wrappable), prefix);
}
}
function safeUnwrap<T>(res: Result<T, T>) {
if(res.isOk()) {
return res.expect("Tried unwrapping message field: " + res)
}
return res.expectErr("Tried unwrapping interaction field" + res)
}

View File

@@ -0,0 +1,18 @@
import { Result as Either } from 'ts-results-es';
import * as assert from 'node:assert';
/**
* @since 3.0.0
*/
export abstract class CoreContext<M, I> {
protected constructor(protected ctx: Either<M, I>) {
assert.ok(typeof ctx === 'object' && ctx != null, "Context was nonobject or null");
}
public isMessage(): this is CoreContext<M, never> {
return this.ctx.isOk();
}
public isSlash(): this is CoreContext<never, I> {
return !this.isMessage();
}
}

View File

@@ -1,20 +0,0 @@
export type Result<Ok, Err> =
| { ok: true; value: Ok }
| { ok: false; error: Err };
export const Ok = <Ok>(value: Ok) => ({ ok: true, value } as const);
export const Err = <Err>(error: Err) => ({ ok: false, error } as const);
export const val = <O, E>(r: Result<O, E>) => r.ok ? r.value : r.error;
export const EMPTY_ERR = Err(undefined);
/**
* Wrap an async operation that may throw an Error (`try-catch` style) into checked exception style
* @param op The operation function
*/
export async function wrapAsync<T, E = unknown>(op: () => Promise<T>): Promise<Result<T, E>> {
try { return op()
.then(Ok)
.catch(Err); }
catch (e) { return Promise.resolve(Err(e as E)); }
}

View File

@@ -8,7 +8,7 @@ import {
import * as Id from '../core/id'
import type { Emitter, ErrorHandling, Logging } from '../core/interfaces';
import { SernError } from '../core/structures/enums'
import { EMPTY_ERR, Err, Ok, Result, wrapAsync } from '../core/structures/result';
import { Err, Ok, Result } from 'ts-results-es';
import type { UnpackedDependencies } from '../types/utility';
import type { CommandModule, Module, Processed } from '../types/core-modules';
import * as assert from 'node:assert';
@@ -17,7 +17,6 @@ import { CommandType } from '../core/structures/enums'
import { inspect } from 'node:util'
import { disposeAll } from '../core/ioc';
import { resultPayload, isAutocomplete, treeSearch, fmt } from '../core/functions'
import merge from 'deepmerge'
function handleError<C>(crashHandler: ErrorHandling, emitter: Emitter, logging?: Logging) {
@@ -44,7 +43,7 @@ interface ExecutePayload {
export const filterTap = <K, R>(onErr: (e: R) => void): OperatorFunction<Result<K, R>, K> =>
concatMap(result => {
if(result.ok){
if(result.isOk()) {
return of(result.value)
}
onErr(result.error);
@@ -143,7 +142,7 @@ export function createInteractionHandler<T extends Interaction>(
.map(({ id, params }) => ({ module: mg.get(id), params }))
.filter(({ module }) => module !== undefined);
if(modules.length == 0) {
return EMPTY_ERR;
return Err.EMPTY;
}
const [{module, params}] = modules;
return Ok(createDispatcher({
@@ -180,9 +179,9 @@ export function createMessageHandler(
* @param task the deferred execution which will be called
*/
export function executeModule(emitter: Emitter, { module, args }: ExecutePayload) {
return from(wrapAsync(async () => module.execute(...args)))
return from(Result.wrapAsync(async () => module.execute(...args)))
.pipe(concatMap(result => {
if (result.ok){
if (result.isOk()) {
emitter.emit('module.activate', resultPayload('success', module));
return EMPTY;
}
@@ -207,10 +206,10 @@ export function createResultResolver<Output>(config: {
return async (payload: ExecutePayload) => {
const task = await callPlugins(payload);
if (!task) throw Error("Plugin did not return anything.");
if(!task.ok) {
onStop?.(payload.module, String(task.error));
} else {
if(task.isOk()) {
return onNext(payload, task.value) as Output;
} else {
onStop?.(payload.module, String(task.error));
}
};
};
@@ -226,13 +225,12 @@ export async function callInitPlugins(_module: Module, deps: Dependencies, emit?
for(const plugin of module.plugins ?? []) {
const result = await plugin.execute({ module, absPath: module.meta.absPath, deps });
if (!result) throw Error("Plugin did not return anything. " + inspect(plugin, false, Infinity, true));
if(!result.ok) {
if(result.isErr()) {
if(emit) {
emitter?.emit('module.register',
resultPayload('failure', module, result.error ?? SernError.PluginFailure));
}
throw Error((result.error ?? SernError.PluginFailure) +
'on module ' + module.name + " " + module.meta.absPath);
throw Error(result.error ?? SernError.PluginFailure);
}
}
return module
@@ -242,7 +240,7 @@ export async function callPlugins({ args, module, deps, params }: ExecutePayload
let state = {};
for(const plugin of module.onEvent??[]) {
const result = await plugin.execute(...args, { state, deps, params, type: module.type });
if(!result.ok) {
if(result.isErr()) {
return result;
}
if(isObject(result.value)) {

View File

@@ -1,4 +1,4 @@
import { concatMap, from, interval, of, map, scan, startWith, fromEvent, take, mergeScan } from "rxjs"
import { concatMap, from, interval, of, map, scan, startWith, fromEvent, take } from "rxjs"
import { Presence } from "../core/presences";
import { Services } from "../core/ioc";
import assert from "node:assert";
@@ -14,7 +14,7 @@ const parseConfig = async (conf: Promise<Presence.Result>) => {
const src$ = typeof repeat === 'number'
? interval(repeat)
: fromEvent(...repeat);
return src$.pipe(mergeScan(async (args) => onRepeat(args), s),
return src$.pipe(scan(onRepeat, s),
startWith(s));
}
return of(s).pipe(take(1));

View File

@@ -107,7 +107,7 @@ export interface ModalSubmitCommand extends Module {
export interface AutocompleteCommand {
onEvent?: ControlPlugin[];
execute: (ctx: AutocompleteInteraction, tbd: SDT) => Awaitable<unknown>;
execute: (ctx: AutocompleteInteraction) => Awaitable<unknown>;
}
export interface DiscordEventCommand<T extends keyof ClientEvents = keyof ClientEvents>
@@ -167,9 +167,9 @@ export interface CommandModuleDefs {
[CommandType.Modal]: ModalSubmitCommand;
}
export interface EventModuleDefs<T extends keyof ClientEvents = keyof ClientEvents> {
export interface EventModuleDefs {
[EventType.Sern]: SernEventCommand;
[EventType.Discord]: DiscordEventCommand<T>;
[EventType.Discord]: DiscordEventCommand;
[EventType.External]: ExternalEventCommand;
}
@@ -186,12 +186,12 @@ export interface SernAutocompleteData
type CommandModuleNoPlugins = {
[T in CommandType]: Omit<CommandModuleDefs[T], 'plugins' | 'onEvent' | 'meta' | 'locals'>;
};
type EventModulesNoPlugins<K extends keyof ClientEvents = keyof ClientEvents> = {
[T in EventType]: Omit<EventModuleDefs<K>[T], 'plugins' | 'onEvent' | 'meta' | 'locals'> ;
type EventModulesNoPlugins = {
[T in EventType]: Omit<EventModuleDefs[T], 'plugins' | 'onEvent' | 'meta' | 'locals'> ;
};
export type InputEvent<K extends keyof ClientEvents = keyof ClientEvents> = {
[T in EventType]: EventModulesNoPlugins<K>[T] & {
export type InputEvent = {
[T in EventType]: EventModulesNoPlugins[T] & {
once?: boolean;
plugins?: InitPlugin[]
};

View File

@@ -11,6 +11,7 @@
* Plugins are reminiscent of middleware in express.
*/
import type { Result } from 'ts-results-es';
import type {
Module,
Processed,
@@ -22,7 +23,6 @@ import type { Context } from '../core/structures/context'
import type {
ButtonInteraction,
ChannelSelectMenuInteraction,
ChatInputCommandInteraction,
MentionableSelectMenuInteraction,
MessageContextMenuCommandInteraction,
ModalSubmitInteraction,
@@ -31,7 +31,6 @@ import type {
UserContextMenuCommandInteraction,
UserSelectMenuInteraction,
} from 'discord.js';
import { Result } from '../core/structures/result';
export type PluginResult = Awaitable<Result<Record<string,unknown>|undefined, string|undefined>>;
export interface InitArgs<T extends Processed<Module> = Processed<Module>> {
@@ -57,8 +56,8 @@ export type AnyPlugin = ControlPlugin | InitPlugin<[InitArgs<Processed<Module>>]
export type CommandArgs<I extends CommandType = CommandType> = CommandArgsMatrix[I]
interface CommandArgsMatrix {
[CommandType.Text]: [Context & { get options(): string[]}, SDT];
[CommandType.Slash]: [Context & { get options(): ChatInputCommandInteraction['options']}, SDT];
[CommandType.Text]: [Context, SDT];
[CommandType.Slash]: [Context, SDT];
[CommandType.Both]: [Context, SDT];
[CommandType.CtxMsg]: [MessageContextMenuCommandInteraction, SDT];
[CommandType.CtxUser]: [UserContextMenuCommandInteraction, SDT];

View File

@@ -1,6 +1,6 @@
import type { InteractionReplyOptions, MessageReplyOptions } from 'discord.js';
import type { Module } from './core-modules';
import type { Result } from '../core/structures/result';
import type { Result } from 'ts-results-es';
export type Awaitable<T> = PromiseLike<T> | T;

View File

@@ -120,6 +120,13 @@ test('init plugins replace array', async () => {
expect(['a']).deep.equal(s.opts)
})
test('call control plugin ', async () => {
const plugin = CommandControlPlugin<CommandType.Slash>((ctx,sdt) => {
return controller.next();
});
const res = await plugin.execute(new ChatInputCommandInteraction(), {})
expect(res.isOk()).toBe(true)
})
test('form sdt', async () => {

View File

@@ -557,6 +557,7 @@ __metadata:
discord.js: ^14.15.3
eslint: 8.39.0
rxjs: ^7.8.0
ts-results-es: ^4.1.0
typescript: 5.0.2
vitest: ^1.6.0
languageName: unknown
@@ -2958,6 +2959,13 @@ __metadata:
languageName: node
linkType: hard
"ts-results-es@npm:^4.1.0":
version: 4.2.0
resolution: "ts-results-es@npm:4.2.0"
checksum: ff475c2f6d44377e0204211e6eafdbcabddf3ad09d40540ad5dee3d817eefbd48c07a21f5ad86864ef82cd8a5542a266af9dd8dd4d58d4766fdd6e79370519bb
languageName: node
linkType: hard
"tslib@npm:2.6.2":
version: 2.6.2
resolution: "tslib@npm:2.6.2"