stonyx 0.2.3-beta.0 → 0.2.3-beta.10
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/.claude/index.md +96 -0
- package/.github/workflows/publish.yml +17 -1
- package/README.md +29 -187
- package/config/environment copy.js +1 -2
- package/config/environment.js +1 -2
- package/docs/api.md +89 -0
- package/docs/cli.md +78 -0
- package/docs/configuration.md +72 -0
- package/docs/conventions/discord-conventions.md +62 -0
- package/docs/conventions/framework-modules.md +127 -0
- package/docs/conventions/index.md +24 -0
- package/docs/conventions/orm-conventions.md +158 -0
- package/docs/conventions/project-structure.md +105 -0
- package/docs/conventions/rest-conventions.md +96 -0
- package/docs/conventions/testing-conventions.md +56 -0
- package/docs/developing-modules.md +108 -0
- package/docs/index.md +15 -0
- package/docs/lifecycle.md +47 -0
- package/docs/logging.md +52 -0
- package/docs/modules.md +78 -0
- package/docs/testing.md +64 -0
- package/package.json +7 -3
- package/scripts/postinstall.js +5 -6
- package/src/cli/help.js +48 -0
- package/src/cli/load-commands.js +56 -0
- package/src/cli/new.js +237 -0
- package/src/cli/serve.js +38 -0
- package/src/cli/test-setup.js +8 -0
- package/src/cli/test.js +27 -0
- package/src/cli.js +63 -0
- package/src/lifecycle.js +17 -0
- package/src/main.js +11 -0
- package/src/bootstrap.cjs +0 -9
- package/stonyx-bootstrap.cjs +0 -9
package/.claude/index.md
ADDED
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
# Stonyx CLI Guide
|
|
2
|
+
|
|
3
|
+
See [docs/index.md](../docs/index.md) for full documentation including conventions, modules, lifecycle, and API reference.
|
|
4
|
+
|
|
5
|
+
## Installation
|
|
6
|
+
|
|
7
|
+
Stonyx is the core framework package. The CLI is provided by the `stonyx` binary defined in `package.json`.
|
|
8
|
+
|
|
9
|
+
In a project that depends on `stonyx`:
|
|
10
|
+
|
|
11
|
+
```bash
|
|
12
|
+
pnpm add -D stonyx
|
|
13
|
+
```
|
|
14
|
+
|
|
15
|
+
The CLI is then available as `npx stonyx` or via pnpm scripts.
|
|
16
|
+
|
|
17
|
+
## Commands
|
|
18
|
+
|
|
19
|
+
### `stonyx serve` (alias: `s`)
|
|
20
|
+
|
|
21
|
+
Bootstrap Stonyx and run the application.
|
|
22
|
+
|
|
23
|
+
```bash
|
|
24
|
+
stonyx serve # loads app.js by default
|
|
25
|
+
stonyx serve --entry custom.js # custom entry point
|
|
26
|
+
```
|
|
27
|
+
|
|
28
|
+
Behavior:
|
|
29
|
+
1. Loads `.env` file
|
|
30
|
+
2. Imports `config/environment.js`
|
|
31
|
+
3. Initializes Stonyx with all detected `@stonyx/*` modules
|
|
32
|
+
4. Runs startup lifecycle hooks
|
|
33
|
+
5. Imports and instantiates the entry point class
|
|
34
|
+
6. Registers SIGTERM/SIGINT handlers for graceful shutdown
|
|
35
|
+
|
|
36
|
+
### `stonyx test` (alias: `t`)
|
|
37
|
+
|
|
38
|
+
Bootstrap Stonyx in test mode and run QUnit tests.
|
|
39
|
+
|
|
40
|
+
```bash
|
|
41
|
+
stonyx test # runs test/**/*-test.js
|
|
42
|
+
stonyx test test/unit/foo-test.js # specific file
|
|
43
|
+
stonyx test "test/integration/**/*-test.js" # glob pattern
|
|
44
|
+
```
|
|
45
|
+
|
|
46
|
+
Sets `NODE_ENV=test` and auto-loads test setup which merges `test/config/environment.js` overrides.
|
|
47
|
+
|
|
48
|
+
### `stonyx help` (alias: `h`)
|
|
49
|
+
|
|
50
|
+
Show available commands including built-in and module-provided commands.
|
|
51
|
+
|
|
52
|
+
```bash
|
|
53
|
+
stonyx help
|
|
54
|
+
stonyx --help
|
|
55
|
+
stonyx -h
|
|
56
|
+
```
|
|
57
|
+
|
|
58
|
+
### `stonyx new <app-name>`
|
|
59
|
+
|
|
60
|
+
Scaffold a new Stonyx project with interactive module selection.
|
|
61
|
+
|
|
62
|
+
```bash
|
|
63
|
+
stonyx new my-backend
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
Prompts for:
|
|
67
|
+
- Package name
|
|
68
|
+
- Which `@stonyx/*` modules to include (REST server, WebSockets, ORM, cron, OAuth, events)
|
|
69
|
+
|
|
70
|
+
Creates:
|
|
71
|
+
- `package.json` with selected dependencies
|
|
72
|
+
- `app.js` entry point
|
|
73
|
+
- `config/environment.js` and `config/environment.example.js`
|
|
74
|
+
- Module-specific directories (`models/`, `requests/`, `crons/`, etc.)
|
|
75
|
+
- `test/` structure with `unit/`, `integration/`, `acceptance/`
|
|
76
|
+
- `.gitignore` and `.nvmrc`
|
|
77
|
+
- Runs `pnpm install`
|
|
78
|
+
|
|
79
|
+
## Module Command System
|
|
80
|
+
|
|
81
|
+
Stonyx modules can register CLI commands by exporting a `./commands` entry in their `package.json` exports map. The CLI auto-discovers these from installed `@stonyx/*` packages.
|
|
82
|
+
|
|
83
|
+
Module commands appear under "Module commands" in `stonyx help` output. They can optionally request Stonyx bootstrap before running (via `bootstrap: true`).
|
|
84
|
+
|
|
85
|
+
## Creating a Stonyx Project Manually
|
|
86
|
+
|
|
87
|
+
If not using `stonyx new`:
|
|
88
|
+
|
|
89
|
+
1. Create project directory
|
|
90
|
+
2. Add `.nvmrc` with current LTS Node version
|
|
91
|
+
3. `pnpm init` and set `"type": "module"`
|
|
92
|
+
4. `pnpm add -D stonyx` plus desired `@stonyx/*` modules
|
|
93
|
+
5. Create `config/environment.js` with module config
|
|
94
|
+
6. Create `app.js` entry point class
|
|
95
|
+
7. Create standard directories per selected modules
|
|
96
|
+
8. Add `@abofs/code-conventions` for linting
|
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
name: Publish to NPM
|
|
2
2
|
|
|
3
3
|
on:
|
|
4
|
+
repository_dispatch:
|
|
5
|
+
types: [cascade-publish]
|
|
4
6
|
workflow_dispatch:
|
|
5
7
|
inputs:
|
|
6
8
|
version-type:
|
|
@@ -17,10 +19,14 @@ on:
|
|
|
17
19
|
type: string
|
|
18
20
|
pull_request:
|
|
19
21
|
types: [opened, synchronize, reopened]
|
|
20
|
-
branches: [main
|
|
22
|
+
branches: [main]
|
|
21
23
|
push:
|
|
22
24
|
branches: [main]
|
|
23
25
|
|
|
26
|
+
concurrency:
|
|
27
|
+
group: ${{ github.event_name == 'repository_dispatch' && 'cascade-update' || format('publish-{0}', github.ref) }}
|
|
28
|
+
cancel-in-progress: false
|
|
29
|
+
|
|
24
30
|
permissions:
|
|
25
31
|
contents: write
|
|
26
32
|
id-token: write
|
|
@@ -28,8 +34,18 @@ permissions:
|
|
|
28
34
|
|
|
29
35
|
jobs:
|
|
30
36
|
publish:
|
|
37
|
+
if: "!contains(github.event.head_commit.message, '[skip ci]')"
|
|
31
38
|
uses: abofs/stonyx-workflows/.github/workflows/npm-publish.yml@main
|
|
32
39
|
with:
|
|
33
40
|
version-type: ${{ github.event.inputs.version-type }}
|
|
34
41
|
custom-version: ${{ github.event.inputs.custom-version }}
|
|
42
|
+
cascade-source: ${{ github.event.client_payload.source_package || '' }}
|
|
43
|
+
secrets: inherit
|
|
44
|
+
|
|
45
|
+
cascade:
|
|
46
|
+
needs: publish
|
|
47
|
+
uses: abofs/stonyx-workflows/.github/workflows/cascade.yml@main
|
|
48
|
+
with:
|
|
49
|
+
package-name: ${{ needs.publish.outputs.package-name }}
|
|
50
|
+
published-version: ${{ needs.publish.outputs.published-version }}
|
|
35
51
|
secrets: inherit
|
package/README.md
CHANGED
|
@@ -1,206 +1,48 @@
|
|
|
1
1
|
# Stonyx
|
|
2
2
|
|
|
3
|
-
**Stonyx** is a lightweight, modular framework for building modern Node.js applications. It provides a
|
|
3
|
+
**Stonyx** is a lightweight, modular framework for building modern Node.js applications. It provides a plug-and-play architecture, centralized color-coded logging, and seamless async module integration.
|
|
4
4
|
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
* ✅ Don’t even hit the ground — just fly
|
|
10
|
-
* ✅ High performance
|
|
11
|
-
|
|
12
|
-
Stonyx acts as a **base application host**, allowing you to add official modules (`@stonyx/*`) or your own custom modules without boilerplate initialization.
|
|
13
|
-
|
|
14
|
-
---
|
|
5
|
+
- 100% JavaScript (ES Modules)
|
|
6
|
+
- Drop-in module system — no boilerplate
|
|
7
|
+
- Automatic async module loading and initialization
|
|
8
|
+
- Built-in CLI for bootstrapping and testing
|
|
15
9
|
|
|
16
10
|
## Quick Start
|
|
17
11
|
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
For standard applications, the **bootstrap file** ensures that the Stonyx framework and all submodules are fully loaded before your application code runs:
|
|
21
|
-
|
|
22
|
-
```js
|
|
23
|
-
// index.js - your project's entry point
|
|
24
|
-
import Stonyx from './stonyx-bootstrap.cjs';
|
|
25
|
-
await Stonyx.ready; // wait until all modules are initialized
|
|
26
|
-
|
|
27
|
-
const { default: App } = await import('./app.js'); // your application
|
|
28
|
-
new App();
|
|
29
|
-
```
|
|
30
|
-
|
|
31
|
-
### CommonJS Bootstrap Helper
|
|
32
|
-
Auto-generated and added to your project post installation
|
|
33
|
-
|
|
34
|
-
```js
|
|
35
|
-
// stonyx-bootstrap.cjs
|
|
36
|
-
const Stonyx = require('stonyx').default;
|
|
37
|
-
const config = require('./config/environment.js').default;
|
|
38
|
-
|
|
39
|
-
new Stonyx(config, __dirname);
|
|
40
|
-
```
|
|
41
|
-
|
|
42
|
-
---
|
|
43
|
-
|
|
44
|
-
## Core Features
|
|
45
|
-
|
|
46
|
-
### 1. **Modular Architecture**
|
|
47
|
-
|
|
48
|
-
* Automatically detects modules in `devDependencies` prefixed with `@stonyx/`.
|
|
49
|
-
* Supports async initialization with `stonyx-async` modules.
|
|
50
|
-
* Safe module sequencing for submodule development with `waitForModule()`.
|
|
51
|
-
|
|
52
|
-
### 2. **Color-Coded Logging**
|
|
53
|
-
|
|
54
|
-
* Centralized logging via [Chronicle](https://github.com/abofs/chronicle).
|
|
55
|
-
* Module-specific logs configurable in `environment.js`:
|
|
56
|
-
|
|
57
|
-
```js
|
|
58
|
-
restServer: { logColor: 'yellow', logMethod: 'api', logTimestamp: true }
|
|
59
|
-
```
|
|
60
|
-
|
|
61
|
-
* Create custom logs for any class with minimal configuration.
|
|
62
|
-
|
|
63
|
-
### 3. **Singleton Design**
|
|
64
|
-
|
|
65
|
-
* Only one instance of Stonyx exists per project.
|
|
66
|
-
* Ensures consistent access to modules, logs, and configuration.
|
|
67
|
-
|
|
68
|
-
### 4. **Plug-and-Play Module Loading**
|
|
69
|
-
|
|
70
|
-
* Modules with `stonyx-module` keyword in `package.json` are auto-initialized.
|
|
71
|
-
* Modules with `stonyx-async` keyword are awaited automatically before usage.
|
|
72
|
-
|
|
73
|
-
---
|
|
74
|
-
|
|
75
|
-
## Official Modules
|
|
76
|
-
|
|
77
|
-
### **[@stonyx/cron](https://github.com/abofs/stonyx-cron)**
|
|
78
|
-
|
|
79
|
-
Lightweight asynchronous job scheduling utility.
|
|
80
|
-
|
|
81
|
-
```js
|
|
82
|
-
import Cron from '@stonyx/cron';
|
|
83
|
-
|
|
84
|
-
const cron = new Cron();
|
|
85
|
-
cron.register('exampleJob', async () => console.log('Job executed!'), 5, true);
|
|
86
|
-
```
|
|
87
|
-
|
|
88
|
-
* Efficient scheduling using a min-heap.
|
|
89
|
-
* Optional logging via `config.cron`.
|
|
90
|
-
|
|
91
|
-
---
|
|
92
|
-
|
|
93
|
-
### **[@stonyx/events](https://github.com/abofs/stonyx-events)**
|
|
94
|
-
|
|
95
|
-
Lightweight pub/sub event system for application-wide event handling.
|
|
96
|
-
|
|
97
|
-
```js
|
|
98
|
-
import Events from '@stonyx/events';
|
|
99
|
-
|
|
100
|
-
const events = new Events();
|
|
101
|
-
|
|
102
|
-
// Register available events
|
|
103
|
-
events.setup(['userLogin', 'userLogout', 'dataChange']);
|
|
104
|
-
|
|
105
|
-
// Subscribe to events
|
|
106
|
-
events.subscribe('userLogin', (user) => {
|
|
107
|
-
console.log(`${user.name} logged in`);
|
|
108
|
-
});
|
|
109
|
-
|
|
110
|
-
// Emit events
|
|
111
|
-
events.emit('userLogin', { name: 'Alice' });
|
|
112
|
-
```
|
|
113
|
-
|
|
114
|
-
* Singleton pattern for shared event bus
|
|
115
|
-
* Async support with error isolation
|
|
116
|
-
* Type-safe event registration
|
|
117
|
-
|
|
118
|
-
---
|
|
119
|
-
|
|
120
|
-
### **[@stonyx/rest-server](https://github.com/abofs/stonyx-rest-server)**
|
|
121
|
-
|
|
122
|
-
Dynamic REST server module with auto-route registration.
|
|
123
|
-
|
|
124
|
-
```js
|
|
125
|
-
import Stonyx from 'stonyx';
|
|
126
|
-
import config from './config/environment.js';
|
|
127
|
-
|
|
128
|
-
new Stonyx(config);
|
|
129
|
-
```
|
|
130
|
-
|
|
131
|
-
* Zero configuration for routes: drop request classes into the `requests` directory.
|
|
132
|
-
* Automatic path generation, JSON parsing, and CORS handling.
|
|
133
|
-
* Supports per-route authentication hooks.
|
|
134
|
-
|
|
135
|
-
---
|
|
136
|
-
|
|
137
|
-
### **[@stonyx/orm](https://github.com/abofs/stonyx-orm)**
|
|
138
|
-
|
|
139
|
-
Lightweight ORM with model definitions, relationships, serializers, and optional REST integration.
|
|
140
|
-
|
|
141
|
-
```js
|
|
142
|
-
import Stonyx from 'stonyx';
|
|
143
|
-
import config from './config/environment.js';
|
|
144
|
-
|
|
145
|
-
new Stonyx(config);
|
|
146
|
-
|
|
147
|
-
// Define models
|
|
148
|
-
import { Model, attr, hasMany, belongsTo } from '@stonyx/orm';
|
|
149
|
-
|
|
150
|
-
class Owner extends Model {
|
|
151
|
-
id = attr('string');
|
|
152
|
-
pets = hasMany('animal');
|
|
153
|
-
}
|
|
154
|
-
```
|
|
155
|
-
|
|
156
|
-
* Auto-loads models, serializers, transforms, and access classes.
|
|
157
|
-
* Optional JSON file persistence with auto-save intervals.
|
|
158
|
-
* Integrates with `@stonyx/rest-server` for automatic route setup.
|
|
159
|
-
|
|
160
|
-
---
|
|
161
|
-
|
|
162
|
-
## Configuration
|
|
163
|
-
|
|
164
|
-
All modules are configurable via `config/environment.js`:
|
|
165
|
-
|
|
166
|
-
```js
|
|
167
|
-
export default {
|
|
168
|
-
restServer: { logColor: 'yellow', port: 3000 },
|
|
169
|
-
orm: { logColor: 'white', db: { file: './db.json', autosave: true } },
|
|
170
|
-
cron: { log: true }
|
|
171
|
-
};
|
|
12
|
+
```bash
|
|
13
|
+
npm install stonyx
|
|
172
14
|
```
|
|
173
15
|
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
## Running the Application
|
|
16
|
+
The CLI handles everything — no manual `new Stonyx()` calls needed:
|
|
177
17
|
|
|
178
18
|
```bash
|
|
179
|
-
|
|
180
|
-
|
|
19
|
+
stonyx serve # Bootstrap + run app.js
|
|
20
|
+
stonyx test # Bootstrap + run tests
|
|
181
21
|
```
|
|
182
22
|
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
## Developing Submodules
|
|
23
|
+
Stonyx reads `config/environment.js`, initializes all `@stonyx/*` modules from your `devDependencies`, and runs your application.
|
|
186
24
|
|
|
187
|
-
|
|
25
|
+
## Documentation
|
|
188
26
|
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
27
|
+
| Section | Description |
|
|
28
|
+
|---------|-------------|
|
|
29
|
+
| [CLI](docs/cli.md) | Commands, aliases, and module commands |
|
|
30
|
+
| [Configuration](docs/configuration.md) | Environment config, module config, test overrides |
|
|
31
|
+
| [Modules](docs/modules.md) | Module architecture, async loading, official modules |
|
|
32
|
+
| [Logging](docs/logging.md) | Chronicle integration and custom log types |
|
|
33
|
+
| [Lifecycle](docs/lifecycle.md) | Startup and shutdown hooks |
|
|
34
|
+
| [Testing](docs/testing.md) | Test runner, helpers, and conventions |
|
|
35
|
+
| [Developing Modules](docs/developing-modules.md) | Guide for building custom Stonyx modules |
|
|
36
|
+
| [API Reference](docs/api.md) | Public exports and class documentation |
|
|
192
37
|
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
// Wait for specific async module readiness
|
|
196
|
-
await waitForModule('restServer');
|
|
197
|
-
```
|
|
198
|
-
|
|
199
|
-
* Use `waitForModule()` **only for submodule development**, testing, or when you need to ensure a specific module is fully initialized before continuing.
|
|
200
|
-
* Official modules automatically initialize during normal application startup, so end-users do **not** need to call `waitForModule()`.
|
|
38
|
+
## Official Modules
|
|
201
39
|
|
|
202
|
-
|
|
40
|
+
| Module | Description |
|
|
41
|
+
|--------|-------------|
|
|
42
|
+
| [@stonyx/cron](https://github.com/abofs/stonyx-cron) | Lightweight async job scheduling |
|
|
43
|
+
| [@stonyx/rest-server](https://github.com/abofs/stonyx-rest-server) | Dynamic REST server with auto-route registration |
|
|
44
|
+
| [@stonyx/orm](https://github.com/abofs/stonyx-orm) | ORM with models, relationships, and serializers |
|
|
203
45
|
|
|
204
46
|
## License
|
|
205
47
|
|
|
206
|
-
Apache 2.0
|
|
48
|
+
Apache 2.0
|
package/config/environment.js
CHANGED
package/docs/api.md
ADDED
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
# API Reference
|
|
2
|
+
|
|
3
|
+
Public exports available from the `stonyx` package.
|
|
4
|
+
|
|
5
|
+
## Exports Map
|
|
6
|
+
|
|
7
|
+
| Import Path | Export | Description |
|
|
8
|
+
|-------------|--------|-------------|
|
|
9
|
+
| `stonyx` | `default` (Stonyx class) | Main framework class (singleton) |
|
|
10
|
+
| `stonyx` | `waitForModule(name)` | Wait for an async module to be ready |
|
|
11
|
+
| `stonyx/config` | `default` (config object) | Live reference to Stonyx configuration |
|
|
12
|
+
| `stonyx/log` | `default` (Chronicle instance) | Live reference to Chronicle logger |
|
|
13
|
+
| `stonyx/lifecycle` | `runStartupHooks(modules)` | Run startup hooks on a module array |
|
|
14
|
+
| `stonyx/lifecycle` | `runShutdownHooks(modules)` | Run shutdown hooks in reverse order |
|
|
15
|
+
| `stonyx/test-helpers` | `setupIntegrationTests(hooks)` | QUnit hook for integration test setup |
|
|
16
|
+
|
|
17
|
+
## Stonyx Class
|
|
18
|
+
|
|
19
|
+
```js
|
|
20
|
+
import Stonyx from 'stonyx';
|
|
21
|
+
```
|
|
22
|
+
|
|
23
|
+
### Constructor
|
|
24
|
+
|
|
25
|
+
```js
|
|
26
|
+
new Stonyx(config, rootPath)
|
|
27
|
+
```
|
|
28
|
+
|
|
29
|
+
- **config** — Full environment configuration object
|
|
30
|
+
- **rootPath** — Absolute path to the project root
|
|
31
|
+
|
|
32
|
+
Returns the existing instance if one already exists (singleton pattern).
|
|
33
|
+
|
|
34
|
+
### Static Properties
|
|
35
|
+
|
|
36
|
+
| Property | Type | Description |
|
|
37
|
+
|----------|------|-------------|
|
|
38
|
+
| `Stonyx.instance` | `Stonyx` | The singleton instance |
|
|
39
|
+
| `Stonyx.ready` | `Promise` | Resolves when all modules are initialized |
|
|
40
|
+
| `Stonyx.initialized` | `boolean` | Whether Stonyx has started initialization |
|
|
41
|
+
|
|
42
|
+
### Static Getters
|
|
43
|
+
|
|
44
|
+
| Getter | Returns | Throws |
|
|
45
|
+
|--------|---------|--------|
|
|
46
|
+
| `Stonyx.config` | Config object | If not initialized |
|
|
47
|
+
| `Stonyx.log` | Chronicle instance | If not initialized |
|
|
48
|
+
|
|
49
|
+
### Instance Properties
|
|
50
|
+
|
|
51
|
+
| Property | Type | Description |
|
|
52
|
+
|----------|------|-------------|
|
|
53
|
+
| `instance.config` | `object` | Merged environment configuration |
|
|
54
|
+
| `instance.chronicle` | `Chronicle` | Logger instance |
|
|
55
|
+
| `instance.modules` | `Array` | Loaded module instances |
|
|
56
|
+
|
|
57
|
+
## waitForModule
|
|
58
|
+
|
|
59
|
+
```js
|
|
60
|
+
import { waitForModule } from 'stonyx';
|
|
61
|
+
|
|
62
|
+
await waitForModule('rest-server');
|
|
63
|
+
```
|
|
64
|
+
|
|
65
|
+
Waits for a specific `@stonyx/*` module to complete initialization. Pass the module name **without** the `@stonyx/` prefix.
|
|
66
|
+
|
|
67
|
+
Throws if the module is not registered in project dependencies.
|
|
68
|
+
|
|
69
|
+
## Lifecycle Functions
|
|
70
|
+
|
|
71
|
+
```js
|
|
72
|
+
import { runStartupHooks, runShutdownHooks } from 'stonyx/lifecycle';
|
|
73
|
+
```
|
|
74
|
+
|
|
75
|
+
### runStartupHooks(modules)
|
|
76
|
+
|
|
77
|
+
Calls `startup()` on each module in array order. Skips modules without a `startup` method.
|
|
78
|
+
|
|
79
|
+
### runShutdownHooks(modules)
|
|
80
|
+
|
|
81
|
+
Calls `shutdown()` on each module in **reverse** array order. Errors are caught and logged — one failing hook does not prevent others from running.
|
|
82
|
+
|
|
83
|
+
## setupIntegrationTests
|
|
84
|
+
|
|
85
|
+
```js
|
|
86
|
+
import { setupIntegrationTests } from 'stonyx/test-helpers';
|
|
87
|
+
```
|
|
88
|
+
|
|
89
|
+
Registers a QUnit `before` hook that waits for `Stonyx.ready`. Use within a `module()` block to ensure Stonyx is fully initialized before tests run.
|
package/docs/cli.md
ADDED
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
# CLI
|
|
2
|
+
|
|
3
|
+
Stonyx includes a CLI that handles bootstrapping, module initialization, and application execution.
|
|
4
|
+
|
|
5
|
+
## Usage
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
stonyx <command> [...args]
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Built-in Commands
|
|
12
|
+
|
|
13
|
+
| Command | Alias | Description |
|
|
14
|
+
|---------|-------|-------------|
|
|
15
|
+
| `serve` | `s` | Bootstrap Stonyx and run the app |
|
|
16
|
+
| `test` | `t` | Bootstrap Stonyx in test mode and run tests |
|
|
17
|
+
| `help` | `h` | Show available commands |
|
|
18
|
+
|
|
19
|
+
### serve
|
|
20
|
+
|
|
21
|
+
Bootstraps Stonyx (loads config, initializes modules, runs lifecycle hooks), then imports your application entry point.
|
|
22
|
+
|
|
23
|
+
```bash
|
|
24
|
+
stonyx serve # Runs app.js by default
|
|
25
|
+
stonyx serve --entry custom.js # Runs a custom entry file
|
|
26
|
+
```
|
|
27
|
+
|
|
28
|
+
The serve command also registers `SIGTERM` and `SIGINT` handlers that run [shutdown hooks](lifecycle.md) before exiting.
|
|
29
|
+
|
|
30
|
+
### test
|
|
31
|
+
|
|
32
|
+
Runs your test suite using [QUnit](https://qunitjs.com/) with automatic Stonyx bootstrapping. Sets `NODE_ENV=test` and applies any [test config overrides](configuration.md#test-environment-overrides).
|
|
33
|
+
|
|
34
|
+
```bash
|
|
35
|
+
stonyx test # Runs test/**/*-test.js by default
|
|
36
|
+
stonyx test "test/unit/**/*.js" # Custom test glob
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
### help
|
|
40
|
+
|
|
41
|
+
Displays all available commands, including any [module commands](#module-commands).
|
|
42
|
+
|
|
43
|
+
```bash
|
|
44
|
+
stonyx help
|
|
45
|
+
```
|
|
46
|
+
|
|
47
|
+
## Module Commands
|
|
48
|
+
|
|
49
|
+
Stonyx modules can register custom CLI commands by exporting a `./commands` entry in their `package.json`. These are automatically discovered and available through the CLI.
|
|
50
|
+
|
|
51
|
+
```json
|
|
52
|
+
{
|
|
53
|
+
"exports": {
|
|
54
|
+
"./commands": "./src/commands.js"
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
```
|
|
58
|
+
|
|
59
|
+
The commands file should export an object mapping command names to definitions:
|
|
60
|
+
|
|
61
|
+
```js
|
|
62
|
+
export default {
|
|
63
|
+
'db:migrate': {
|
|
64
|
+
description: 'Run database migrations',
|
|
65
|
+
bootstrap: true,
|
|
66
|
+
run: async ({ args, cwd }) => { /* ... */ }
|
|
67
|
+
}
|
|
68
|
+
};
|
|
69
|
+
```
|
|
70
|
+
|
|
71
|
+
- **`bootstrap: true`** — Stonyx will be fully initialized before the command runs
|
|
72
|
+
- **`bootstrap: false`** — The command runs without Stonyx initialization
|
|
73
|
+
|
|
74
|
+
Module commands appear under "Module commands" in `stonyx help` output. If two modules register the same command name, the first one loaded wins and a warning is printed.
|
|
75
|
+
|
|
76
|
+
## Environment Variables
|
|
77
|
+
|
|
78
|
+
The CLI automatically loads `.env` files via `process.loadEnvFile()` before executing any command.
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
# Configuration
|
|
2
|
+
|
|
3
|
+
Stonyx uses a centralized configuration file that all modules read from at startup.
|
|
4
|
+
|
|
5
|
+
## Environment Config
|
|
6
|
+
|
|
7
|
+
Your project's configuration lives at `config/environment.js`:
|
|
8
|
+
|
|
9
|
+
```js
|
|
10
|
+
const { DEBUG, NODE_ENV } = process.env;
|
|
11
|
+
|
|
12
|
+
const environment = NODE_ENV ?? 'development';
|
|
13
|
+
|
|
14
|
+
export default {
|
|
15
|
+
environment,
|
|
16
|
+
debug: DEBUG ?? environment === 'development',
|
|
17
|
+
|
|
18
|
+
// Module-specific configuration
|
|
19
|
+
restServer: { logColor: 'yellow', port: 3000 },
|
|
20
|
+
cron: { log: true },
|
|
21
|
+
};
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
This file is auto-generated on `npm install` via the postinstall script if it doesn't already exist.
|
|
25
|
+
|
|
26
|
+
> **Note:** `config/environment.js` is gitignored by default so each environment can have its own settings.
|
|
27
|
+
|
|
28
|
+
## Module Configuration
|
|
29
|
+
|
|
30
|
+
Each Stonyx module reads its configuration from a top-level key matching its camelCase name. For example, `@stonyx/rest-server` reads from `config.restServer`.
|
|
31
|
+
|
|
32
|
+
Async modules ship with their own default config at `config/environment.js` inside the module package. Your project config is merged on top of these defaults — you only need to specify overrides.
|
|
33
|
+
|
|
34
|
+
## Logging Configuration
|
|
35
|
+
|
|
36
|
+
Any config key with a `logColor` property automatically creates a [Chronicle](logging.md) log type:
|
|
37
|
+
|
|
38
|
+
```js
|
|
39
|
+
export default {
|
|
40
|
+
myService: {
|
|
41
|
+
logColor: 'purple', // Required — enables log creation
|
|
42
|
+
logMethod: 'highlight', // Optional — custom method name (defaults to key name)
|
|
43
|
+
logTimestamp: true, // Optional — include timestamps
|
|
44
|
+
},
|
|
45
|
+
};
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
This works for both module configs and custom service configs. See [Logging](logging.md) for details.
|
|
49
|
+
|
|
50
|
+
## Test Environment Overrides
|
|
51
|
+
|
|
52
|
+
When `NODE_ENV=test`, Stonyx automatically looks for `test/config/environment.js` in your project root:
|
|
53
|
+
|
|
54
|
+
```js
|
|
55
|
+
// test/config/environment.js
|
|
56
|
+
export default {
|
|
57
|
+
debug: false,
|
|
58
|
+
restServer: { port: 0 },
|
|
59
|
+
};
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
These overrides are deep-merged into the main config using in-place mutation, so any existing references (like `stonyx/config` exports) stay valid.
|
|
63
|
+
|
|
64
|
+
## Accessing Config at Runtime
|
|
65
|
+
|
|
66
|
+
```js
|
|
67
|
+
import config from 'stonyx/config';
|
|
68
|
+
|
|
69
|
+
console.log(config.environment); // 'development', 'test', etc.
|
|
70
|
+
```
|
|
71
|
+
|
|
72
|
+
The `stonyx/config` export is a live reference to the Stonyx instance config. It will throw if accessed before Stonyx is initialized.
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
# Discord Conventions
|
|
2
|
+
|
|
3
|
+
Conventions for projects using `@stonyx/discord`.
|
|
4
|
+
|
|
5
|
+
## Command Files
|
|
6
|
+
|
|
7
|
+
- One class per file in the configured command directory (default: `discord-commands/`)
|
|
8
|
+
- Each file default-exports a class extending `Command` from `@stonyx/discord`
|
|
9
|
+
- Class must define a `data` property (a `SlashCommandBuilder` instance) and an `async execute(interaction)` method
|
|
10
|
+
- Class ordering: static properties → `data` → `execute()`
|
|
11
|
+
- Filenames use kebab-case; `forEachFileImport` converts to camelCase for internal registration
|
|
12
|
+
- Permission checks belong in the consuming application, not in the module
|
|
13
|
+
|
|
14
|
+
```js
|
|
15
|
+
import { Command } from '@stonyx/discord';
|
|
16
|
+
import { SlashCommandBuilder } from 'discord.js';
|
|
17
|
+
|
|
18
|
+
export default class PingCommand extends Command {
|
|
19
|
+
data = new SlashCommandBuilder()
|
|
20
|
+
.setName('ping')
|
|
21
|
+
.setDescription('Replies with Pong!');
|
|
22
|
+
|
|
23
|
+
async execute(interaction) {
|
|
24
|
+
await interaction.reply('Pong!');
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
```
|
|
28
|
+
|
|
29
|
+
## Event Handler Files
|
|
30
|
+
|
|
31
|
+
- One class per file in the configured event directory (default: `discord-events/`)
|
|
32
|
+
- Each file default-exports a class extending `EventHandler` from `@stonyx/discord`
|
|
33
|
+
- Class must set `static event` to a valid Discord.js event name (e.g., `'messageCreate'`, `'voiceStateUpdate'`)
|
|
34
|
+
- Class must implement a `handle(...args)` method matching the Discord.js event signature
|
|
35
|
+
- Class ordering: static properties → `handle()`
|
|
36
|
+
|
|
37
|
+
```js
|
|
38
|
+
import { EventHandler } from '@stonyx/discord';
|
|
39
|
+
|
|
40
|
+
export default class WelcomeHandler extends EventHandler {
|
|
41
|
+
static event = 'guildMemberAdd';
|
|
42
|
+
|
|
43
|
+
handle(member) {
|
|
44
|
+
// Welcome new member
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
```
|
|
48
|
+
|
|
49
|
+
## Configuration
|
|
50
|
+
|
|
51
|
+
- Config namespace: `config.discord`
|
|
52
|
+
- Environment variables prefixed with `DISCORD_`
|
|
53
|
+
- Logging: `log.discord()` — configured via `logColor: '#7289da'` and `logMethod: 'discord'`
|
|
54
|
+
- Intent auto-derivation: the module computes required gateway intents from discovered event handlers; use `additionalIntents` in config for edge cases
|
|
55
|
+
|
|
56
|
+
## Module Keywords
|
|
57
|
+
|
|
58
|
+
```json
|
|
59
|
+
"keywords": ["stonyx-async", "stonyx-module"]
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
`stonyx-async` indicates the module's `init()` is async and must be awaited by the framework.
|