@clipr/worker 0.0.8 → 0.0.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.
Files changed (35) hide show
  1. package/CHANGELOG.md +14 -0
  2. package/package.json +2 -2
  3. package/.dev.vars +0 -2
  4. package/coverage/base.css +0 -224
  5. package/coverage/block-navigation.js +0 -87
  6. package/coverage/clover.xml +0 -432
  7. package/coverage/coverage-final.json +0 -16
  8. package/coverage/favicon.png +0 -0
  9. package/coverage/index.html +0 -161
  10. package/coverage/prettify.css +0 -1
  11. package/coverage/prettify.js +0 -2
  12. package/coverage/sort-arrow-sprite.png +0 -0
  13. package/coverage/sorter.js +0 -210
  14. package/coverage/src/crypto.ts.html +0 -340
  15. package/coverage/src/index.html +0 -176
  16. package/coverage/src/index.ts.html +0 -262
  17. package/coverage/src/kv-backend.ts.html +0 -295
  18. package/coverage/src/kv.ts.html +0 -577
  19. package/coverage/src/middleware/auth.ts.html +0 -250
  20. package/coverage/src/middleware/index.html +0 -116
  21. package/coverage/src/routes/health.ts.html +0 -100
  22. package/coverage/src/routes/import-export.ts.html +0 -232
  23. package/coverage/src/routes/index.html +0 -221
  24. package/coverage/src/routes/links.ts.html +0 -379
  25. package/coverage/src/routes/password.ts.html +0 -436
  26. package/coverage/src/routes/qr.ts.html +0 -202
  27. package/coverage/src/routes/redirect.ts.html +0 -346
  28. package/coverage/src/routes/shorten.ts.html +0 -340
  29. package/coverage/src/routes/stats.ts.html +0 -145
  30. package/coverage/src/test-utils.ts.html +0 -184
  31. package/coverage/src/utils/index.html +0 -116
  32. package/coverage/src/utils/qr.ts.html +0 -190
  33. package/dist/README.md +0 -1
  34. package/dist/index.js +0 -7139
  35. package/dist/index.js.map +0 -8
@@ -1,340 +0,0 @@
1
-
2
- <!doctype html>
3
- <html lang="en">
4
-
5
- <head>
6
- <title>Code coverage report for src/routes/shorten.ts</title>
7
- <meta charset="utf-8" />
8
- <link rel="stylesheet" href="../../prettify.css" />
9
- <link rel="stylesheet" href="../../base.css" />
10
- <link rel="shortcut icon" type="image/x-icon" href="../../favicon.png" />
11
- <meta name="viewport" content="width=device-width, initial-scale=1" />
12
- <style type='text/css'>
13
- .coverage-summary .sorter {
14
- background-image: url(../../sort-arrow-sprite.png);
15
- }
16
- </style>
17
- </head>
18
-
19
- <body>
20
- <div class='wrapper'>
21
- <div class='pad1'>
22
- <h1><a href="../../index.html">All files</a> / <a href="index.html">src/routes</a> shorten.ts</h1>
23
- <div class='clearfix'>
24
-
25
- <div class='fl pad1y space-right2'>
26
- <span class="strong">96% </span>
27
- <span class="quiet">Statements</span>
28
- <span class='fraction'>24/25</span>
29
- </div>
30
-
31
-
32
- <div class='fl pad1y space-right2'>
33
- <span class="strong">78.57% </span>
34
- <span class="quiet">Branches</span>
35
- <span class='fraction'>22/28</span>
36
- </div>
37
-
38
-
39
- <div class='fl pad1y space-right2'>
40
- <span class="strong">100% </span>
41
- <span class="quiet">Functions</span>
42
- <span class='fraction'>1/1</span>
43
- </div>
44
-
45
-
46
- <div class='fl pad1y space-right2'>
47
- <span class="strong">96% </span>
48
- <span class="quiet">Lines</span>
49
- <span class='fraction'>24/25</span>
50
- </div>
51
-
52
-
53
- </div>
54
- <p class="quiet">
55
- Press <em>n</em> or <em>j</em> to go to the next uncovered block, <em>b</em>, <em>p</em> or <em>k</em> for the previous block.
56
- </p>
57
- <template id="filterTemplate">
58
- <div class="quiet">
59
- Filter:
60
- <input type="search" id="fileSearch">
61
- </div>
62
- </template>
63
- </div>
64
- <div class='status-line high'></div>
65
- <pre><table class="coverage">
66
- <tr><td class="line-count quiet"><a name='L1'></a><a href='#L1'>1</a>
67
- <a name='L2'></a><a href='#L2'>2</a>
68
- <a name='L3'></a><a href='#L3'>3</a>
69
- <a name='L4'></a><a href='#L4'>4</a>
70
- <a name='L5'></a><a href='#L5'>5</a>
71
- <a name='L6'></a><a href='#L6'>6</a>
72
- <a name='L7'></a><a href='#L7'>7</a>
73
- <a name='L8'></a><a href='#L8'>8</a>
74
- <a name='L9'></a><a href='#L9'>9</a>
75
- <a name='L10'></a><a href='#L10'>10</a>
76
- <a name='L11'></a><a href='#L11'>11</a>
77
- <a name='L12'></a><a href='#L12'>12</a>
78
- <a name='L13'></a><a href='#L13'>13</a>
79
- <a name='L14'></a><a href='#L14'>14</a>
80
- <a name='L15'></a><a href='#L15'>15</a>
81
- <a name='L16'></a><a href='#L16'>16</a>
82
- <a name='L17'></a><a href='#L17'>17</a>
83
- <a name='L18'></a><a href='#L18'>18</a>
84
- <a name='L19'></a><a href='#L19'>19</a>
85
- <a name='L20'></a><a href='#L20'>20</a>
86
- <a name='L21'></a><a href='#L21'>21</a>
87
- <a name='L22'></a><a href='#L22'>22</a>
88
- <a name='L23'></a><a href='#L23'>23</a>
89
- <a name='L24'></a><a href='#L24'>24</a>
90
- <a name='L25'></a><a href='#L25'>25</a>
91
- <a name='L26'></a><a href='#L26'>26</a>
92
- <a name='L27'></a><a href='#L27'>27</a>
93
- <a name='L28'></a><a href='#L28'>28</a>
94
- <a name='L29'></a><a href='#L29'>29</a>
95
- <a name='L30'></a><a href='#L30'>30</a>
96
- <a name='L31'></a><a href='#L31'>31</a>
97
- <a name='L32'></a><a href='#L32'>32</a>
98
- <a name='L33'></a><a href='#L33'>33</a>
99
- <a name='L34'></a><a href='#L34'>34</a>
100
- <a name='L35'></a><a href='#L35'>35</a>
101
- <a name='L36'></a><a href='#L36'>36</a>
102
- <a name='L37'></a><a href='#L37'>37</a>
103
- <a name='L38'></a><a href='#L38'>38</a>
104
- <a name='L39'></a><a href='#L39'>39</a>
105
- <a name='L40'></a><a href='#L40'>40</a>
106
- <a name='L41'></a><a href='#L41'>41</a>
107
- <a name='L42'></a><a href='#L42'>42</a>
108
- <a name='L43'></a><a href='#L43'>43</a>
109
- <a name='L44'></a><a href='#L44'>44</a>
110
- <a name='L45'></a><a href='#L45'>45</a>
111
- <a name='L46'></a><a href='#L46'>46</a>
112
- <a name='L47'></a><a href='#L47'>47</a>
113
- <a name='L48'></a><a href='#L48'>48</a>
114
- <a name='L49'></a><a href='#L49'>49</a>
115
- <a name='L50'></a><a href='#L50'>50</a>
116
- <a name='L51'></a><a href='#L51'>51</a>
117
- <a name='L52'></a><a href='#L52'>52</a>
118
- <a name='L53'></a><a href='#L53'>53</a>
119
- <a name='L54'></a><a href='#L54'>54</a>
120
- <a name='L55'></a><a href='#L55'>55</a>
121
- <a name='L56'></a><a href='#L56'>56</a>
122
- <a name='L57'></a><a href='#L57'>57</a>
123
- <a name='L58'></a><a href='#L58'>58</a>
124
- <a name='L59'></a><a href='#L59'>59</a>
125
- <a name='L60'></a><a href='#L60'>60</a>
126
- <a name='L61'></a><a href='#L61'>61</a>
127
- <a name='L62'></a><a href='#L62'>62</a>
128
- <a name='L63'></a><a href='#L63'>63</a>
129
- <a name='L64'></a><a href='#L64'>64</a>
130
- <a name='L65'></a><a href='#L65'>65</a>
131
- <a name='L66'></a><a href='#L66'>66</a>
132
- <a name='L67'></a><a href='#L67'>67</a>
133
- <a name='L68'></a><a href='#L68'>68</a>
134
- <a name='L69'></a><a href='#L69'>69</a>
135
- <a name='L70'></a><a href='#L70'>70</a>
136
- <a name='L71'></a><a href='#L71'>71</a>
137
- <a name='L72'></a><a href='#L72'>72</a>
138
- <a name='L73'></a><a href='#L73'>73</a>
139
- <a name='L74'></a><a href='#L74'>74</a>
140
- <a name='L75'></a><a href='#L75'>75</a>
141
- <a name='L76'></a><a href='#L76'>76</a>
142
- <a name='L77'></a><a href='#L77'>77</a>
143
- <a name='L78'></a><a href='#L78'>78</a>
144
- <a name='L79'></a><a href='#L79'>79</a>
145
- <a name='L80'></a><a href='#L80'>80</a>
146
- <a name='L81'></a><a href='#L81'>81</a>
147
- <a name='L82'></a><a href='#L82'>82</a>
148
- <a name='L83'></a><a href='#L83'>83</a>
149
- <a name='L84'></a><a href='#L84'>84</a>
150
- <a name='L85'></a><a href='#L85'>85</a>
151
- <a name='L86'></a><a href='#L86'>86</a></td><td class="line-coverage quiet"><span class="cline-any cline-neutral">&nbsp;</span>
152
- <span class="cline-any cline-neutral">&nbsp;</span>
153
- <span class="cline-any cline-neutral">&nbsp;</span>
154
- <span class="cline-any cline-neutral">&nbsp;</span>
155
- <span class="cline-any cline-neutral">&nbsp;</span>
156
- <span class="cline-any cline-neutral">&nbsp;</span>
157
- <span class="cline-any cline-neutral">&nbsp;</span>
158
- <span class="cline-any cline-neutral">&nbsp;</span>
159
- <span class="cline-any cline-neutral">&nbsp;</span>
160
- <span class="cline-any cline-neutral">&nbsp;</span>
161
- <span class="cline-any cline-neutral">&nbsp;</span>
162
- <span class="cline-any cline-neutral">&nbsp;</span>
163
- <span class="cline-any cline-neutral">&nbsp;</span>
164
- <span class="cline-any cline-neutral">&nbsp;</span>
165
- <span class="cline-any cline-neutral">&nbsp;</span>
166
- <span class="cline-any cline-neutral">&nbsp;</span>
167
- <span class="cline-any cline-neutral">&nbsp;</span>
168
- <span class="cline-any cline-neutral">&nbsp;</span>
169
- <span class="cline-any cline-neutral">&nbsp;</span>
170
- <span class="cline-any cline-neutral">&nbsp;</span>
171
- <span class="cline-any cline-neutral">&nbsp;</span>
172
- <span class="cline-any cline-neutral">&nbsp;</span>
173
- <span class="cline-any cline-neutral">&nbsp;</span>
174
- <span class="cline-any cline-yes">8x</span>
175
- <span class="cline-any cline-yes">8x</span>
176
- <span class="cline-any cline-neutral">&nbsp;</span>
177
- <span class="cline-any cline-yes">1x</span>
178
- <span class="cline-any cline-neutral">&nbsp;</span>
179
- <span class="cline-any cline-neutral">&nbsp;</span>
180
- <span class="cline-any cline-yes">7x</span>
181
- <span class="cline-any cline-yes">1x</span>
182
- <span class="cline-any cline-neutral">&nbsp;</span>
183
- <span class="cline-any cline-neutral">&nbsp;</span>
184
- <span class="cline-any cline-neutral">&nbsp;</span>
185
- <span class="cline-any cline-yes">6x</span>
186
- <span class="cline-any cline-yes">6x</span>
187
- <span class="cline-any cline-yes">1x</span>
188
- <span class="cline-any cline-neutral">&nbsp;</span>
189
- <span class="cline-any cline-neutral">&nbsp;</span>
190
- <span class="cline-any cline-neutral">&nbsp;</span>
191
- <span class="cline-any cline-neutral">&nbsp;</span>
192
- <span class="cline-any cline-yes">5x</span>
193
- <span class="cline-any cline-yes">4x</span>
194
- <span class="cline-any cline-yes">4x</span>
195
- <span class="cline-any cline-yes">1x</span>
196
- <span class="cline-any cline-neutral">&nbsp;</span>
197
- <span class="cline-any cline-yes">3x</span>
198
- <span class="cline-any cline-neutral">&nbsp;</span>
199
- <span class="cline-any cline-neutral">&nbsp;</span>
200
- <span class="cline-any cline-yes">3x</span>
201
- <span class="cline-any cline-yes">3x</span>
202
- <span class="cline-any cline-yes">1x</span>
203
- <span class="cline-any cline-neutral">&nbsp;</span>
204
- <span class="cline-any cline-neutral">&nbsp;</span>
205
- <span class="cline-any cline-neutral">&nbsp;</span>
206
- <span class="cline-any cline-yes">1x</span>
207
- <span class="cline-any cline-yes">1x</span>
208
- <span class="cline-any cline-neutral">&nbsp;</span>
209
- <span class="cline-any cline-neutral">&nbsp;</span>
210
- <span class="cline-any cline-neutral">&nbsp;</span>
211
- <span class="cline-any cline-neutral">&nbsp;</span>
212
- <span class="cline-any cline-yes">3x</span>
213
- <span class="cline-any cline-no">&nbsp;</span>
214
- <span class="cline-any cline-neutral">&nbsp;</span>
215
- <span class="cline-any cline-neutral">&nbsp;</span>
216
- <span class="cline-any cline-yes">3x</span>
217
- <span class="cline-any cline-yes">3x</span>
218
- <span class="cline-any cline-neutral">&nbsp;</span>
219
- <span class="cline-any cline-neutral">&nbsp;</span>
220
- <span class="cline-any cline-neutral">&nbsp;</span>
221
- <span class="cline-any cline-neutral">&nbsp;</span>
222
- <span class="cline-any cline-neutral">&nbsp;</span>
223
- <span class="cline-any cline-neutral">&nbsp;</span>
224
- <span class="cline-any cline-neutral">&nbsp;</span>
225
- <span class="cline-any cline-neutral">&nbsp;</span>
226
- <span class="cline-any cline-neutral">&nbsp;</span>
227
- <span class="cline-any cline-neutral">&nbsp;</span>
228
- <span class="cline-any cline-neutral">&nbsp;</span>
229
- <span class="cline-any cline-neutral">&nbsp;</span>
230
- <span class="cline-any cline-neutral">&nbsp;</span>
231
- <span class="cline-any cline-yes">8x</span>
232
- <span class="cline-any cline-neutral">&nbsp;</span>
233
- <span class="cline-any cline-yes">3x</span>
234
- <span class="cline-any cline-yes">3x</span>
235
- <span class="cline-any cline-neutral">&nbsp;</span>
236
- <span class="cline-any cline-neutral">&nbsp;</span></td><td class="text"><pre class="prettyprint lang-js">import type { ShortUrl, UtmParams } from '@clipr/core';
237
- import { generateSlug, validateSlug, validateUrl } from '@clipr/core';
238
- import type { Context } from 'hono';
239
- import { hashPassword } from '../crypto.js';
240
- import { getUrl, incrementCounter, putUrl } from '../kv.js';
241
- import type { Env } from '../types.js';
242
- &nbsp;
243
- interface ShortenBody {
244
- url: string;
245
- slug?: string;
246
- tags?: string[];
247
- expiresAt?: string;
248
- password?: string;
249
- description?: string;
250
- utm?: UtmParams;
251
- ogTitle?: string;
252
- ogDescription?: string;
253
- ogImage?: string;
254
- }
255
- &nbsp;
256
- /** POST /api/shorten — Create a new short URL. */
257
- export async function handleShorten(c: Context&lt;{ Bindings: Env }&gt;): Promise&lt;Response&gt; {
258
- let body: ShortenBody;
259
- try {
260
- body = await c.req.json&lt;ShortenBody&gt;();
261
- } catch {
262
- return c.json({ error: 'Invalid JSON body' }, 400);
263
- }
264
- &nbsp;
265
- if (!body.url) {
266
- return c.json({ error: 'Missing required field: url' }, 400);
267
- }
268
- &nbsp;
269
- // Validate URL
270
- const urlResult = validateUrl(body.url);
271
- if (!urlResult.valid) {
272
- return c.json({ error: `Invalid URL: ${urlResult.reason}` }, 400);
273
- }
274
- &nbsp;
275
- // Determine slug
276
- let slug: string;
277
- if (body.slug) {
278
- const slugResult = validateSlug(body.slug);
279
- if (!slugResult.valid) {
280
- return c.json({ error: `Invalid slug: ${slugResult.reason}` }, 400);
281
- }
282
- slug = body.slug;
283
- &nbsp;
284
- // Check for conflicts
285
- const existing = await getUrl(c.env.URLS, slug);
286
- if (existing) {
287
- return c.json({ error: `Slug "${slug}" already exists` }, 409);
288
- }
289
- } else {
290
- // Auto-generate slug from counter
291
- const counter = await incrementCounter(c.env.URLS);
292
- slug = generateSlug(counter);
293
- }
294
- &nbsp;
295
- // Hash password if provided
296
- let passwordHash: string | undefined;
297
- <span class="missing-if-branch" title="if path not taken" >I</span>if (body.password) {
298
- <span class="cstat-no" title="statement not covered" > passwordHash = await hashPassword(body.password);</span>
299
- }
300
- &nbsp;
301
- const now = new Date().toISOString();
302
- const entry: ShortUrl = {
303
- slug,
304
- url: body.url,
305
- createdAt: now,
306
- ...(body.expiresAt &amp;&amp; <span class="branch-1 cbranch-no" title="branch not covered" >{ expiresAt: body.expiresAt }),</span>
307
- ...(body.description &amp;&amp; { description: body.description }),
308
- ...(body.tags &amp;&amp; { tags: body.tags }),
309
- ...(body.utm &amp;&amp; { utm: body.utm }),
310
- ...(passwordHash &amp;&amp; <span class="branch-1 cbranch-no" title="branch not covered" >{ passwordHash }),</span>
311
- ...(body.ogTitle &amp;&amp; <span class="branch-1 cbranch-no" title="branch not covered" >{ ogTitle: body.ogTitle }),</span>
312
- ...(body.ogDescription &amp;&amp; <span class="branch-1 cbranch-no" title="branch not covered" >{ ogDescription: body.ogDescription }),</span>
313
- ...(body.ogImage &amp;&amp; <span class="branch-1 cbranch-no" title="branch not covered" >{ ogImage: body.ogImage }),</span>
314
- };
315
- &nbsp;
316
- await putUrl(c.env.URLS, slug, entry);
317
- &nbsp;
318
- const shortUrl = `${c.env.BASE_URL}/${slug}`;
319
- return c.json({ slug, shortUrl, url: body.url }, 201);
320
- }
321
- &nbsp;</pre></td></tr></table></pre>
322
-
323
- <div class='push'></div><!-- for sticky footer -->
324
- </div><!-- /wrapper -->
325
- <div class='footer quiet pad2 space-top1 center small'>
326
- Code coverage generated by
327
- <a href="https://istanbul.js.org/" target="_blank" rel="noopener noreferrer">istanbul</a>
328
- at 2026-03-31T14:33:55.154Z
329
- </div>
330
- <script src="../../prettify.js"></script>
331
- <script>
332
- window.onload = function () {
333
- prettyPrint();
334
- };
335
- </script>
336
- <script src="../../sorter.js"></script>
337
- <script src="../../block-navigation.js"></script>
338
- </body>
339
- </html>
340
-
@@ -1,145 +0,0 @@
1
-
2
- <!doctype html>
3
- <html lang="en">
4
-
5
- <head>
6
- <title>Code coverage report for src/routes/stats.ts</title>
7
- <meta charset="utf-8" />
8
- <link rel="stylesheet" href="../../prettify.css" />
9
- <link rel="stylesheet" href="../../base.css" />
10
- <link rel="shortcut icon" type="image/x-icon" href="../../favicon.png" />
11
- <meta name="viewport" content="width=device-width, initial-scale=1" />
12
- <style type='text/css'>
13
- .coverage-summary .sorter {
14
- background-image: url(../../sort-arrow-sprite.png);
15
- }
16
- </style>
17
- </head>
18
-
19
- <body>
20
- <div class='wrapper'>
21
- <div class='pad1'>
22
- <h1><a href="../../index.html">All files</a> / <a href="index.html">src/routes</a> stats.ts</h1>
23
- <div class='clearfix'>
24
-
25
- <div class='fl pad1y space-right2'>
26
- <span class="strong">87.5% </span>
27
- <span class="quiet">Statements</span>
28
- <span class='fraction'>7/8</span>
29
- </div>
30
-
31
-
32
- <div class='fl pad1y space-right2'>
33
- <span class="strong">75% </span>
34
- <span class="quiet">Branches</span>
35
- <span class='fraction'>3/4</span>
36
- </div>
37
-
38
-
39
- <div class='fl pad1y space-right2'>
40
- <span class="strong">100% </span>
41
- <span class="quiet">Functions</span>
42
- <span class='fraction'>1/1</span>
43
- </div>
44
-
45
-
46
- <div class='fl pad1y space-right2'>
47
- <span class="strong">87.5% </span>
48
- <span class="quiet">Lines</span>
49
- <span class='fraction'>7/8</span>
50
- </div>
51
-
52
-
53
- </div>
54
- <p class="quiet">
55
- Press <em>n</em> or <em>j</em> to go to the next uncovered block, <em>b</em>, <em>p</em> or <em>k</em> for the previous block.
56
- </p>
57
- <template id="filterTemplate">
58
- <div class="quiet">
59
- Filter:
60
- <input type="search" id="fileSearch">
61
- </div>
62
- </template>
63
- </div>
64
- <div class='status-line high'></div>
65
- <pre><table class="coverage">
66
- <tr><td class="line-count quiet"><a name='L1'></a><a href='#L1'>1</a>
67
- <a name='L2'></a><a href='#L2'>2</a>
68
- <a name='L3'></a><a href='#L3'>3</a>
69
- <a name='L4'></a><a href='#L4'>4</a>
70
- <a name='L5'></a><a href='#L5'>5</a>
71
- <a name='L6'></a><a href='#L6'>6</a>
72
- <a name='L7'></a><a href='#L7'>7</a>
73
- <a name='L8'></a><a href='#L8'>8</a>
74
- <a name='L9'></a><a href='#L9'>9</a>
75
- <a name='L10'></a><a href='#L10'>10</a>
76
- <a name='L11'></a><a href='#L11'>11</a>
77
- <a name='L12'></a><a href='#L12'>12</a>
78
- <a name='L13'></a><a href='#L13'>13</a>
79
- <a name='L14'></a><a href='#L14'>14</a>
80
- <a name='L15'></a><a href='#L15'>15</a>
81
- <a name='L16'></a><a href='#L16'>16</a>
82
- <a name='L17'></a><a href='#L17'>17</a>
83
- <a name='L18'></a><a href='#L18'>18</a>
84
- <a name='L19'></a><a href='#L19'>19</a>
85
- <a name='L20'></a><a href='#L20'>20</a>
86
- <a name='L21'></a><a href='#L21'>21</a></td><td class="line-coverage quiet"><span class="cline-any cline-neutral">&nbsp;</span>
87
- <span class="cline-any cline-neutral">&nbsp;</span>
88
- <span class="cline-any cline-neutral">&nbsp;</span>
89
- <span class="cline-any cline-neutral">&nbsp;</span>
90
- <span class="cline-any cline-neutral">&nbsp;</span>
91
- <span class="cline-any cline-neutral">&nbsp;</span>
92
- <span class="cline-any cline-yes">3x</span>
93
- <span class="cline-any cline-yes">3x</span>
94
- <span class="cline-any cline-no">&nbsp;</span>
95
- <span class="cline-any cline-neutral">&nbsp;</span>
96
- <span class="cline-any cline-neutral">&nbsp;</span>
97
- <span class="cline-any cline-neutral">&nbsp;</span>
98
- <span class="cline-any cline-yes">3x</span>
99
- <span class="cline-any cline-yes">3x</span>
100
- <span class="cline-any cline-yes">1x</span>
101
- <span class="cline-any cline-neutral">&nbsp;</span>
102
- <span class="cline-any cline-neutral">&nbsp;</span>
103
- <span class="cline-any cline-yes">2x</span>
104
- <span class="cline-any cline-yes">2x</span>
105
- <span class="cline-any cline-neutral">&nbsp;</span>
106
- <span class="cline-any cline-neutral">&nbsp;</span></td><td class="text"><pre class="prettyprint lang-js">import type { Context } from 'hono';
107
- import { getStats, getUrl } from '../kv.js';
108
- import type { Env } from '../types.js';
109
- &nbsp;
110
- /** GET /api/stats/:code — Returns LinkStats JSON for a link. */
111
- export async function handleStats(c: Context&lt;{ Bindings: Env }&gt;): Promise&lt;Response&gt; {
112
- const code = c.req.param('code');
113
- <span class="missing-if-branch" title="if path not taken" >I</span>if (!code) {
114
- <span class="cstat-no" title="statement not covered" > return c.json({ error: 'Missing code parameter' }, 400);</span>
115
- }
116
- &nbsp;
117
- // Verify the link exists
118
- const entry = await getUrl(c.env.URLS, code);
119
- if (!entry) {
120
- return c.json({ error: `Link "${code}" not found` }, 404);
121
- }
122
- &nbsp;
123
- const stats = await getStats(c.env.URLS, code);
124
- return c.json({ code, ...stats });
125
- }
126
- &nbsp;</pre></td></tr></table></pre>
127
-
128
- <div class='push'></div><!-- for sticky footer -->
129
- </div><!-- /wrapper -->
130
- <div class='footer quiet pad2 space-top1 center small'>
131
- Code coverage generated by
132
- <a href="https://istanbul.js.org/" target="_blank" rel="noopener noreferrer">istanbul</a>
133
- at 2026-03-31T14:33:55.154Z
134
- </div>
135
- <script src="../../prettify.js"></script>
136
- <script>
137
- window.onload = function () {
138
- prettyPrint();
139
- };
140
- </script>
141
- <script src="../../sorter.js"></script>
142
- <script src="../../block-navigation.js"></script>
143
- </body>
144
- </html>
145
-