-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.ts
122 lines (108 loc) · 3.05 KB
/
index.ts
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
import Counter from './counter';
import verifyAuthorizationToken from './authentication';
function generateJSONResponse(
body: Record<string, unknown>,
status = 200,
additionalHeaders: Record<string, unknown> = {}
) {
return new Response(JSON.stringify(body), {
status,
headers: {
'Content-Type': 'application/json',
...additionalHeaders,
},
});
}
function getCounterObjectStub(
pathname: string,
counterNamespace: DurableObjectNamespace<Counter>
) {
const counterID: DurableObjectId = counterNamespace.idFromName(pathname);
return counterNamespace.get(counterID);
}
async function handleGetRequest(
pathname: string,
counterNamespace: DurableObjectNamespace<Counter>
) {
const counterObjectStub = getCounterObjectStub(pathname, counterNamespace);
const newValue = await counterObjectStub.increment();
return generateJSONResponse({ newValue });
}
async function handleDeleteRequest(
pathname: string,
counterNamespace: DurableObjectNamespace<Counter>,
authorizationToken: string,
expectedAuthorizationTokenHash: string
) {
const authorizationTokenValidity = await verifyAuthorizationToken(
authorizationToken,
expectedAuthorizationTokenHash
);
if (!authorizationTokenValidity) {
return generateJSONResponse(
{ message: 'The request is not authorized.' },
401
);
}
const counterObjectStub = getCounterObjectStub(pathname, counterNamespace);
await counterObjectStub.destroy();
return generateJSONResponse({
message: 'The counter is scheduled to be destroyed.',
});
}
async function handleRequest(
pathname: string,
method: string,
counterNamespace: DurableObjectNamespace<Counter>,
authorizationToken: string,
expectedAuthorizationTokenHash: string
) {
switch (method) {
case 'GET':
return await handleGetRequest(pathname, counterNamespace);
case 'DELETE':
return await handleDeleteRequest(
pathname,
counterNamespace,
authorizationToken,
expectedAuthorizationTokenHash
);
default:
return generateJSONResponse(
{ message: 'The method used in this request is not supported.' },
405,
{
Allow: 'GET, DELETE',
}
);
}
}
export default {
async fetch(request, env): Promise<Response> {
const { COUNTER: COUNTER_NAMESPACE, DELETE_TOKEN_HASH } = env;
const { url, method, headers } = request;
const { pathname } = new URL(url);
const authorizationToken = (headers.get('Authorization') ?? '').replace(
'Bearer ',
''
);
try {
return await handleRequest(
pathname,
method,
COUNTER_NAMESPACE,
authorizationToken,
DELETE_TOKEN_HASH
);
} catch (error) {
if (error instanceof Error && error.message.startsWith('InvalidState')) {
return generateJSONResponse({ message: error.message }, 400);
}
return generateJSONResponse(
{ message: 'Unknown internal server error occured.' },
500
);
}
},
} satisfies ExportedHandler<Env>;
export { Counter };