-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathapp.js
144 lines (126 loc) · 3.4 KB
/
app.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
import Fastify from "fastify";
import AutoLoad from "@fastify/autoload";
import Env from "@fastify/env";
import pino from "pino";
import pretty from "pino-pretty";
import { join } from "desm";
import fs from "node:fs";
import path from "node:path";
import { fileURLToPath } from "node:url";
/**
* Configuration
*/
// Environment schema
const envSchema = {
type: "object",
required: ["POSTGRES_URL", "JWT_SECRET"],
properties: {
POSTGRES_URL: { type: "string" },
JWT_SECRET: { type: "string" },
NODE_ENV: { type: "string", default: "development" },
PORT: { type: "string", default: "3000" },
},
};
// Logger configuration
const createLogger = () => {
const stream = pretty({
translateTime: "SYS:HH:MM:ss Z",
messageFormat: "{msg} {req.method} {req.url}",
include: "time,pid,level",
hideObject: true,
colorize: process.env.NODE_ENV === "development",
});
return pino({ level: "info" }, stream);
};
/**
* Authentication middleware
*/
const createAuthenticator = (app) => {
// Decorate with authenticate method but don't apply globally
app.decorate("authenticate", async (request, reply) => {
try {
await request.jwtVerify();
} catch (err) {
reply.code(401);
return reply.send({ error: "Unauthorized", message: err.message });
}
});
};
/**
* Route registration
*/
const registerRoutes = async (app) => {
const routesDir = path.join(path.dirname(fileURLToPath(import.meta.url)), "routes");
// Register main routes file
await app.register(import("./routes/routes.js"));
// Dynamically register all subdirectory route files
const subdirs = fs.readdirSync(routesDir, { withFileTypes: true })
.filter(dirent => dirent.isDirectory())
.map(dirent => dirent.name);
for (const dir of subdirs) {
const routeFile = `./routes/${dir}/routes.js`;
try {
await app.register(import(routeFile));
app.log.info(`Registered routes from ${routeFile}`);
} catch (error) {
app.log.error(`Failed to register routes from ${routeFile}: ${error.message}`);
}
}
};
/**
* Server initialization
*/
const buildApp = async () => {
// Initialize Fastify with logger
const loggerInstance = createLogger();
const app = Fastify({
loggerInstance,
trustProxy: true
});
// Register environment variables
await app.register(Env, {
confKey: "secrets",
schema: envSchema,
dotenv: true,
});
// Setup authentication
//createAuthenticator(app);
// Register plugins
await app.register(AutoLoad, {
dir: join(import.meta.url, "plugins"),
dirNameRoutePrefix: false,
ignorePattern: /.*.no-load\.js/,
indexPattern: /^no$/i,
});
// Register routes
await registerRoutes(app);
return app;
};
/**
* Server startup
*/
const startServer = async (app) => {
try {
const port = app.secrets.PORT || "3000";
const isDev = app.secrets.NODE_ENV === "development";
const host = isDev ? "127.0.0.1" : "0.0.0.0";
await app.listen({ host, port });
app.log.info(`Server running in ${app.secrets.NODE_ENV} mode on http://${host}:${port}`);
} catch (error) {
app.log.error(`Error starting server: ${error.message}`);
process.exit(1);
}
};
/**
* Main execution
*/
const main = async () => {
try {
const app = await buildApp();
await startServer(app);
} catch (error) {
console.error("Fatal error during startup:", error);
process.exit(1);
}
};
main();