forked from live-codes/livecodes
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.ts
113 lines (101 loc) · 3.57 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
/// <reference path="../node_modules/@cloudflare/workers-types/index.d.ts" />
import { getProjectInfo } from './utils';
type Env = Record<'API_TOKEN', string>;
type Data = Record<string, unknown>;
type PgFunction = PagesFunction<Env, 'id', Data>;
type Context = EventContext<Env, 'id', Data>;
export const onRequest: PgFunction = async function (context) {
const { request, env } = context;
const originalResponse = await env.ASSETS.fetch(request);
const cf = (request as any).cf;
// server-side analytics
const data = {
url: request.url,
resource: 'app',
method: request.method,
date: String(new Date()),
colo: cf?.colo,
country: cf?.country,
httpProtocol: cf?.httpProtocol,
city: cf?.city,
continent: cf?.continent,
region: cf?.region,
regionCode: cf?.regionCode,
timezone: cf?.timezone,
accept: request.headers.get('accept'),
'accept-encoding': request.headers.get('accept-encoding'),
'accept-language': request.headers.get('accept-language'),
referer: request.headers.get('referer'),
'sec-ch-ua': request.headers.get('sec-ch-ua'),
'sec-ch-ua-mobile': request.headers.get('sec-ch-ua-mobile'),
'sec-ch-ua-platform': request.headers.get('sec-ch-ua-platform'),
'sec-fetch-dest': request.headers.get('sec-fetch-dest'),
'sec-fetch-mode': request.headers.get('sec-fetch-mode'),
'sec-fetch-site': request.headers.get('sec-fetch-site'),
'user-agent': request.headers.get('user-agent'),
};
try {
// oEmbed & meta tags
const url = new URL(request.url);
const oembedUrl = encodeURIComponent(url.href);
const { title, description } = await getProjectInfo(url);
const modifiedBody = (await originalResponse.text())
.replace(
`href="oembed?url=https%3A%2F%2Flivecodes.io&format=json"`,
`href="${url.origin}/oembed?url=${oembedUrl}&format=json"`,
)
.replace(
/title" content="LiveCodes"/g,
`title" content="${
!title || title === 'Untitled Project' ? 'LiveCodes' : title + ' - LiveCodes'
}"`,
)
.replace(
/content="Code Playground That Just Works!"/g,
`content="${
!title && !description
? 'Code Playground That Just Works!'
: description || 'A project on LiveCodes.'
}"`,
)
.replace(/content="https:\/\/livecodes.io\/"/g, `content="${request.url}"`)
.replace(/content="https:\/\/livecodes.io\/livecodes/g, `content="${url.origin}/livecodes`);
const response = new Response(modifiedBody, originalResponse);
const linkHeader = `<${url.origin}/oembed?url=${oembedUrl}&format=json>; rel="alternate"; type="application/json+oembed"; title="LiveCodes"`;
response.headers.append('Link', linkHeader);
context.data = {
...data,
ok: response.ok,
'content-encoding': response.headers.get('content-encoding'),
'content-type': response.headers.get('content-type'),
status: response.status,
statusText: response.statusText,
};
context.waitUntil(logToAPI(context));
return response;
} catch (err) {
context.data = {
...data,
ok: false,
error: err.message || err,
};
context.waitUntil(logToAPI(context));
return originalResponse;
}
};
export const logToAPI = (context: Context) => {
const { data, env } = context;
return fetch('https://api2.livecodes.io/log', {
method: 'POST',
headers: {
'API-Token': env.API_TOKEN,
'Content-Type': 'application/json',
},
body: JSON.stringify({
type: 'analytics',
data,
}),
}).catch(() => {
//
});
};