Initial commit
This commit is contained in:
commit
666e298635
20 changed files with 3951 additions and 0 deletions
18
.eslintrc.yml
Normal file
18
.eslintrc.yml
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
env:
|
||||||
|
es2020: true
|
||||||
|
worker: true
|
||||||
|
extends: ["plugin:@typescript-eslint/recommended"]
|
||||||
|
parser: "@typescript-eslint/parser"
|
||||||
|
parserOptions:
|
||||||
|
ecmaVersion: 2020
|
||||||
|
ecmaFeatures:
|
||||||
|
impliedStrict: true
|
||||||
|
sourceType: module
|
||||||
|
root: true
|
||||||
|
rules:
|
||||||
|
indent: [error, 2, { SwitchCase: 1 }]
|
||||||
|
semi: [error, always]
|
||||||
|
quotes: [error, single, avoid-escape]
|
||||||
|
no-trailing-spaces: [error]
|
||||||
|
no-unused-vars: [error, { argsIgnorePattern: ^_ }]
|
||||||
|
prefer-const: [error, { destructuring: all }]
|
18
.github/workflows/deploy-master-to-prod.yml
vendored
Normal file
18
.github/workflows/deploy-master-to-prod.yml
vendored
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
name: Deploy master on push
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
workflow_dispatch:
|
||||||
|
jobs:
|
||||||
|
deploy:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: wrangler publish
|
||||||
|
uses: cloudflare/wrangler-action@1.3.0
|
||||||
|
with:
|
||||||
|
apiToken: ${{ secrets.CF_API_TOKEN }}
|
||||||
|
environment: omaps
|
||||||
|
|
||||||
|
# TODO: Add organicmaps deploy.
|
3
.gitignore
vendored
Normal file
3
.gitignore
vendored
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
# Autogenerated by esbuild.
|
||||||
|
workers-site/index.js
|
||||||
|
node_modules/
|
1
.nvmrc
Normal file
1
.nvmrc
Normal file
|
@ -0,0 +1 @@
|
||||||
|
v14.17.0
|
7
.prettierrc
Normal file
7
.prettierrc
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
{
|
||||||
|
"printWidth": 120,
|
||||||
|
"tabWidth": 2,
|
||||||
|
"singleQuote": true,
|
||||||
|
"trailingComma": "all",
|
||||||
|
"arrowParens": "always"
|
||||||
|
}
|
52
README.md
Normal file
52
README.md
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
# Static resources and short links (ge0) decoder for Organic Maps
|
||||||
|
|
||||||
|
Root domain redirects to https://organicmaps.app/.
|
||||||
|
|
||||||
|
URLs like `http(s)://omaps.app/ENCODEDCOORDINATES/PINNAME` are decoded to lat, lon and zoom level. Then the OSM
|
||||||
|
map is displayed and url schemes are opened on mobile apps.
|
||||||
|
|
||||||
|
Add some query parameters to test:
|
||||||
|
|
||||||
|
- For dev environment:
|
||||||
|
- [http](http://url-processor.omaps.workers.dev/B4srhdHVVt/Some+Name)
|
||||||
|
- [https](https://url-processor.omaps.workers.dev/B4srhdHVVt/Some+Name)
|
||||||
|
- [http ru](http://url-processor.omaps.workers.dev/AwAAAAAAAA/%d0%9c%d0%b8%d0%bd%d1%81%d0%ba_%d1%83%d0%bb._%d0%9b%d0%b5%d0%bd%d0%b8%d0%bd%d0%b0_9)
|
||||||
|
- [https ru](https://url-processor.omaps.workers.dev/AwAAAAAAAA/%d0%9c%d0%b8%d0%bd%d1%81%d0%ba_%d1%83%d0%bb._%d0%9b%d0%b5%d0%bd%d0%b8%d0%bd%d0%b0_9)
|
||||||
|
- For prod environment:
|
||||||
|
- [http](http://omaps.app/B4srhdHVVt/Some+Name)
|
||||||
|
- [https](https://omaps.app/B4srhdHVVt/Some+Name)
|
||||||
|
- [http ru](http://omaps.app/AwAAAAAAAA/%d0%9c%d0%b8%d0%bd%d1%81%d0%ba_%d1%83%d0%bb._%d0%9b%d0%b5%d0%bd%d0%b8%d0%bd%d0%b0_9)
|
||||||
|
- [https ru](https://omaps.app/AwAAAAAAAA/%d0%9c%d0%b8%d0%bd%d1%81%d0%ba_%d1%83%d0%bb._%d0%9b%d0%b5%d0%bd%d0%b8%d0%bd%d0%b0_9)
|
||||||
|
|
||||||
|
[](https://deploy.workers.cloudflare.com/?url=https://github.com/organicmaps/url-processor)
|
||||||
|
|
||||||
|
## Requirements
|
||||||
|
|
||||||
|
Install CloudFlare's wrangler and other dev dependencies using npm:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
npm i
|
||||||
|
```
|
||||||
|
|
||||||
|
## Development
|
||||||
|
|
||||||
|
Use `npx wrangler dev` for localhost development.
|
||||||
|
|
||||||
|
## Preview on workers.dev
|
||||||
|
|
||||||
|
Use `npx wrangler preview` to open and test deployed worker in browser.
|
||||||
|
|
||||||
|
## Deployment
|
||||||
|
|
||||||
|
All pushes to master automatically deploy dev version to https://url-processor.omaps.workers.dev/
|
||||||
|
|
||||||
|
Deploy to prod manually using `npx wrangler publish --env omaps` or this
|
||||||
|
[action](https://github.com/organicmaps/url-processor/actions/workflows/deploy-master-to-prod.yml).
|
||||||
|
|
||||||
|
## Known issues
|
||||||
|
|
||||||
|
- Cloudflare's free Flexible SSL certificates does not support 4-th level
|
||||||
|
subdomains like a.b.example.com, so you can see strange SSL errors.
|
||||||
|
- HTTPS `fetch` requests from Workers are converted to HTTP ones if the target
|
||||||
|
host is in the same Cloudflare zone, see [here](https://community.cloudflare.com/t/does-cloudflare-worker-allow-secure-https-connection-to-fetch-even-on-flexible-ssl/68051/12)
|
||||||
|
for more details.
|
1
metadata.json
Normal file
1
metadata.json
Normal file
|
@ -0,0 +1 @@
|
||||||
|
{ "version": 1 }
|
3407
package-lock.json
generated
Normal file
3407
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load diff
27
package.json
Normal file
27
package.json
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
{
|
||||||
|
"private": true,
|
||||||
|
"name": "omaps-url-handler",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "Processes specific HTTP requests to domains",
|
||||||
|
"main": "dist/index.js",
|
||||||
|
"scripts": {
|
||||||
|
"build": "esbuild src/index.ts --bundle --outfile=workers-site/index.js",
|
||||||
|
"test": "eslint src/**/*.ts && tsc --noEmit",
|
||||||
|
"format": "prettier --write 'src/**/*.{ts,tsx,json}'"
|
||||||
|
},
|
||||||
|
"author": "Alexander Borsuk <me@alex.bio>",
|
||||||
|
"license": "MIT",
|
||||||
|
"devDependencies": {
|
||||||
|
"@cloudflare/workers-types": "^2.2.2",
|
||||||
|
"@cloudflare/wrangler": "^1.16.1",
|
||||||
|
"@typescript-eslint/eslint-plugin": "^4.24.0",
|
||||||
|
"@typescript-eslint/parser": "^4.24.0",
|
||||||
|
"esbuild": "^0.12.1",
|
||||||
|
"eslint": "^7.26.0",
|
||||||
|
"prettier": "^2.3.0",
|
||||||
|
"typescript": "^4.2.4"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@cloudflare/kv-asset-handler": "^0.1.2"
|
||||||
|
}
|
||||||
|
}
|
28
public/.well-known/apple-app-site-association
Normal file
28
public/.well-known/apple-app-site-association
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
{
|
||||||
|
"applinks": {
|
||||||
|
"apps": [],
|
||||||
|
"details": [
|
||||||
|
{
|
||||||
|
"appID": "9Z6432XD7L.app.organicmaps",
|
||||||
|
"paths": [
|
||||||
|
"NOT /",
|
||||||
|
"*"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"appID": "9Z6432XD7L.app.organicmaps.debug",
|
||||||
|
"paths": [
|
||||||
|
"NOT /",
|
||||||
|
"*"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"appID": "9Z6432XD7L.app.organicmaps.beta",
|
||||||
|
"paths": [
|
||||||
|
"NOT /",
|
||||||
|
"*"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
38
public/.well-known/assetlinks.json
Normal file
38
public/.well-known/assetlinks.json
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"relation": [
|
||||||
|
"delegate_permission/common.handle_all_urls"
|
||||||
|
],
|
||||||
|
"target": {
|
||||||
|
"namespace": "android_app",
|
||||||
|
"package_name": "app.organicmaps",
|
||||||
|
"sha256_cert_fingerprints": [
|
||||||
|
"B9:C7:AE:79:A5:A9:02:70:DF:08:A1:32:E5:36:B9:C6:66:F5:BE:F1:F5:9B:30:4F:CE:CF:86:87:86:5E:4B:5B"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"relation": [
|
||||||
|
"delegate_permission/common.handle_all_urls"
|
||||||
|
],
|
||||||
|
"target": {
|
||||||
|
"namespace": "android_app",
|
||||||
|
"package_name": "app.organicmaps.beta",
|
||||||
|
"sha256_cert_fingerprints": [
|
||||||
|
"B9:C7:AE:79:A5:A9:02:70:DF:08:A1:32:E5:36:B9:C6:66:F5:BE:F1:F5:9B:30:4F:CE:CF:86:87:86:5E:4B:5B"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"relation": [
|
||||||
|
"delegate_permission/common.handle_all_urls"
|
||||||
|
],
|
||||||
|
"target": {
|
||||||
|
"namespace": "android_app",
|
||||||
|
"package_name": "app.organicmaps.debug",
|
||||||
|
"sha256_cert_fingerprints": [
|
||||||
|
"96:D6:28:81:78:B0:1B:86:9B:D3:FF:BF:95:B3:3B:EE:DE:23:01:68:DF:88:2A:1D:7A:4B:B2:8B:85:34:59:F4"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
1
public/apple-app-site-association
Symbolic link
1
public/apple-app-site-association
Symbolic link
|
@ -0,0 +1 @@
|
||||||
|
./.well-known/apple-app-site-association
|
77
public/ge0.html
Normal file
77
public/ge0.html
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
<!-- Template which is processed and returned by worker. -->
|
||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<link rel="stylesheet" href="https://unpkg.com/leaflet@1.7.1/dist/leaflet.css" integrity="sha512-xodZBNTC5n17Xt2atTPuE1HxjVMSvLVW9ocqUKLsCC5CXdbqCmblAshOMAS6/keqq/sMZMZ19scR4PsZChSR7A==" crossorigin="" />
|
||||||
|
<style>
|
||||||
|
html,
|
||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
background-color: green;
|
||||||
|
}
|
||||||
|
|
||||||
|
a,
|
||||||
|
a:link,
|
||||||
|
a:visited,
|
||||||
|
a:hover,
|
||||||
|
a:active {
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
#header {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
#map {
|
||||||
|
height: 400px;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<script src="https://unpkg.com/leaflet@1.7.1/dist/leaflet.js" integrity="sha512-XQoYMqMTK8LvdxXYG3nZ448hOEQiglfqkJs1NOQV44cWnUrBc8PkAOcXy20w0vlaXaVUearIOBhiXZ5V3ynxwA==" crossorigin=""></script>
|
||||||
|
<title>${title}</title>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<div id="header">
|
||||||
|
<h2 id="name">${name}</h2>
|
||||||
|
<h3>
|
||||||
|
<a id="open" href="#" onclick="onOpenClick">Open in <strong>O</strong>rganic <strong>M</strong>aps app</a>
|
||||||
|
<a href="https://organicmaps.app/">Install <strong>O</strong>rganic <strong>M</strong>aps app</a>
|
||||||
|
<a href="https://www.openstreetmap.org/?mlat=${lat}&mlon=${lon}#map=${zoom}/${lat}/${lon}" target="_blank">See on OpenStreet Map</a>
|
||||||
|
</h3>
|
||||||
|
</div>
|
||||||
|
<div id="map" class="map"></div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
const isiOS = navigator.platform.substr(0, 2) === 'iP' || // iPhone, iPad, iPod, including simulators.
|
||||||
|
(navigator.userAgent.includes('Mac') && 'ontouchend' in document); // iPad on iOS 13.
|
||||||
|
const isAndroid = !isiOS && /(android)/i.test(navigator.userAgent);
|
||||||
|
const isDesktop = !isiOS && !isAndroid;
|
||||||
|
|
||||||
|
if (isDesktop)
|
||||||
|
document.getElementById('open').remove();
|
||||||
|
|
||||||
|
function onOpenClick() {
|
||||||
|
window.open('om://' + location.pathname);
|
||||||
|
}
|
||||||
|
const lat = ${ lat };
|
||||||
|
const lon = ${ lon };
|
||||||
|
const zoom = ${ zoom };
|
||||||
|
const map = L.map('map').setView([lat, lon], zoom);
|
||||||
|
L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
|
||||||
|
attribution: '© <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors',
|
||||||
|
maxZoom: 19,
|
||||||
|
}).addTo(map);
|
||||||
|
|
||||||
|
const marker = L.marker([lat, lon]).addTo(map);
|
||||||
|
marker.bindPopup('${name}');//.openPopup();
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
1
public/robots.txt
Normal file
1
public/robots.txt
Normal file
|
@ -0,0 +1 @@
|
||||||
|
User-agent: *
|
6
src/bindings.d.ts
vendored
Normal file
6
src/bindings.d.ts
vendored
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
export { };
|
||||||
|
|
||||||
|
// Defined in wrangler.toml
|
||||||
|
declare global {
|
||||||
|
const DEBUG: boolean;
|
||||||
|
}
|
131
src/ge0.ts
Normal file
131
src/ge0.ts
Normal file
|
@ -0,0 +1,131 @@
|
||||||
|
declare type LatLonZoom = {
|
||||||
|
lat: number;
|
||||||
|
lon: number;
|
||||||
|
zoom: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
function replaceInTemplate(template: string, data: Record<string, any>) {
|
||||||
|
const pattern = /\${\s*(\w+?)\s*}/g;
|
||||||
|
return template.replace(pattern, (_, token) => data[token] || '');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Throws on decode error.
|
||||||
|
export async function onGe0Decode(template: string, encodedLatLonZoom: string, name?: string): Promise<Response> {
|
||||||
|
const llz = decodeLatLonZoom(encodedLatLonZoom);
|
||||||
|
let title = 'Organic Maps';
|
||||||
|
if (name) {
|
||||||
|
name = decodeURIComponent(name.replaceAll('_', ' ')).replaceAll('"', "'");
|
||||||
|
title = name + ' | ' + title;
|
||||||
|
} else {
|
||||||
|
name = '♥';
|
||||||
|
}
|
||||||
|
|
||||||
|
template = replaceInTemplate(template, { ...llz, title, name });
|
||||||
|
return new Response(template, { headers: { 'content-type': 'text/html' } });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Throws exceptions on errors.
|
||||||
|
function decodeLatLonZoom(encodedLatLonZoom: string): LatLonZoom {
|
||||||
|
const GE0_MAX_POINT_BYTES = 10;
|
||||||
|
const GE0_MAX_COORD_BITS = GE0_MAX_POINT_BYTES * 3;
|
||||||
|
|
||||||
|
let zoom = base64Reverse[encodedLatLonZoom.charCodeAt(0)];
|
||||||
|
if (zoom > 63) throw new Error('Invalid zoom level: the url was not encoded properly');
|
||||||
|
zoom = Math.round(zoom / 4 + 4);
|
||||||
|
|
||||||
|
const latLonStr = encodedLatLonZoom.substr(1);
|
||||||
|
const latLonBytes = latLonStr.length;
|
||||||
|
|
||||||
|
let lat = 0;
|
||||||
|
let lon = 0;
|
||||||
|
|
||||||
|
for (let i = 0, shift = GE0_MAX_COORD_BITS - 3; i < latLonBytes; i++, shift -= 3) {
|
||||||
|
const a = base64Reverse[latLonStr.charCodeAt(i)];
|
||||||
|
const lat1 = (((a >> 5) & 1) << 2) | (((a >> 3) & 1) << 1) | ((a >> 1) & 1);
|
||||||
|
const lon1 = (((a >> 4) & 1) << 2) | (((a >> 2) & 1) << 1) | (a & 1);
|
||||||
|
lat |= lat1 << shift;
|
||||||
|
lon |= lon1 << shift;
|
||||||
|
}
|
||||||
|
|
||||||
|
const middleOfSquare = 1 << (3 * (GE0_MAX_POINT_BYTES - latLonBytes) - 1);
|
||||||
|
lat += middleOfSquare;
|
||||||
|
lon += middleOfSquare;
|
||||||
|
|
||||||
|
lat = (lat / ((1 << GE0_MAX_COORD_BITS) - 1)) * 180.0 - 90.0;
|
||||||
|
lon = (lon / (1 << GE0_MAX_COORD_BITS)) * 360.0 - 180.0;
|
||||||
|
|
||||||
|
lat = Math.round(lat * 1e5) / 1e5;
|
||||||
|
lon = Math.round(lon * 1e5) / 1e5;
|
||||||
|
|
||||||
|
if (lat <= -90.0 || lat >= 90.0 || lon <= -180.0 || lon >= 180.0)
|
||||||
|
throw new Error('Invalid coordinates: the url was not encoded properly');
|
||||||
|
|
||||||
|
return { lat, lon, zoom };
|
||||||
|
}
|
||||||
|
|
||||||
|
const base64Reverse: { [key: number]: number } = {
|
||||||
|
65: 0,
|
||||||
|
66: 1,
|
||||||
|
67: 2,
|
||||||
|
68: 3,
|
||||||
|
69: 4,
|
||||||
|
70: 5,
|
||||||
|
71: 6,
|
||||||
|
72: 7,
|
||||||
|
73: 8,
|
||||||
|
74: 9,
|
||||||
|
75: 10,
|
||||||
|
76: 11,
|
||||||
|
77: 12,
|
||||||
|
78: 13,
|
||||||
|
79: 14,
|
||||||
|
80: 15,
|
||||||
|
81: 16,
|
||||||
|
82: 17,
|
||||||
|
83: 18,
|
||||||
|
84: 19,
|
||||||
|
85: 20,
|
||||||
|
86: 21,
|
||||||
|
87: 22,
|
||||||
|
88: 23,
|
||||||
|
89: 24,
|
||||||
|
90: 25,
|
||||||
|
97: 26,
|
||||||
|
98: 27,
|
||||||
|
99: 28,
|
||||||
|
100: 29,
|
||||||
|
101: 30,
|
||||||
|
102: 31,
|
||||||
|
103: 32,
|
||||||
|
104: 33,
|
||||||
|
105: 34,
|
||||||
|
106: 35,
|
||||||
|
107: 36,
|
||||||
|
108: 37,
|
||||||
|
109: 38,
|
||||||
|
110: 39,
|
||||||
|
111: 40,
|
||||||
|
112: 41,
|
||||||
|
113: 42,
|
||||||
|
114: 43,
|
||||||
|
115: 44,
|
||||||
|
116: 45,
|
||||||
|
117: 46,
|
||||||
|
118: 47,
|
||||||
|
119: 48,
|
||||||
|
120: 49,
|
||||||
|
121: 50,
|
||||||
|
122: 51,
|
||||||
|
48: 52,
|
||||||
|
49: 53,
|
||||||
|
50: 54,
|
||||||
|
51: 55,
|
||||||
|
52: 56,
|
||||||
|
53: 57,
|
||||||
|
54: 58,
|
||||||
|
55: 59,
|
||||||
|
56: 60,
|
||||||
|
57: 61,
|
||||||
|
45: 62,
|
||||||
|
95: 63,
|
||||||
|
};
|
54
src/index.ts
Normal file
54
src/index.ts
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
import { getAssetFromKV, mapRequestToAsset } from '@cloudflare/kv-asset-handler';
|
||||||
|
import { onGe0Decode } from './ge0';
|
||||||
|
|
||||||
|
const NOT_FOUND_REDIRECT_URL = 'https://organicmaps.app/';
|
||||||
|
const GE0_TEMPLATE_URL = '/ge0.html';
|
||||||
|
|
||||||
|
addEventListener('fetch', (event) => {
|
||||||
|
try {
|
||||||
|
event.respondWith(handleFetchEvent(event));
|
||||||
|
} catch (e) {
|
||||||
|
if (DEBUG) {
|
||||||
|
event.respondWith(new Response(e.message || e.toString(), { status: 500 }));
|
||||||
|
} else {
|
||||||
|
// In case of unexpected errors, always redirect to the default url.
|
||||||
|
event.respondWith(onRedirect(NOT_FOUND_REDIRECT_URL));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
async function handleFetchEvent(event: FetchEvent) {
|
||||||
|
// See https://github.com/cloudflare/kv-asset-handler#optional-arguments
|
||||||
|
const getAssetOptions: {
|
||||||
|
cacheControl?: { bypassCache: boolean };
|
||||||
|
mapRequestToAsset?: (_: Request) => Request;
|
||||||
|
} = {};
|
||||||
|
|
||||||
|
if (DEBUG) {
|
||||||
|
// Disable caching in debug.
|
||||||
|
getAssetOptions.cacheControl = { bypassCache: true };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try to return a static resource first.
|
||||||
|
try {
|
||||||
|
return await getAssetFromKV(event, getAssetOptions);
|
||||||
|
} catch (_) { }
|
||||||
|
// No static resource were found, try to handle a specific dynamic request.
|
||||||
|
const { pathname } = new URL(event.request.url);
|
||||||
|
// Filter empty pathname elements.
|
||||||
|
const params = pathname.split('/').filter(Boolean);
|
||||||
|
if (params.length === 0) return onRedirect(NOT_FOUND_REDIRECT_URL);
|
||||||
|
|
||||||
|
getAssetOptions.mapRequestToAsset = (request: Request) => {
|
||||||
|
const url = new URL(request.url);
|
||||||
|
url.pathname = GE0_TEMPLATE_URL;
|
||||||
|
return mapRequestToAsset(new Request(url.toString(), request));
|
||||||
|
};
|
||||||
|
const resp = await getAssetFromKV(event, getAssetOptions);
|
||||||
|
const ge0HtmlTemplate = await resp.text();
|
||||||
|
return onGe0Decode(ge0HtmlTemplate, params[0], params.length >= 2 ? params[1] : undefined);
|
||||||
|
}
|
||||||
|
|
||||||
|
function onRedirect(url: string) {
|
||||||
|
return Response.redirect(url, 301);
|
||||||
|
}
|
28
tsconfig.json
Normal file
28
tsconfig.json
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"isolatedModules": true,
|
||||||
|
"outDir": "./dist",
|
||||||
|
"module": "ES6",
|
||||||
|
"moduleResolution": "node",
|
||||||
|
"target": "ES2021",
|
||||||
|
"lib": [
|
||||||
|
"ES2021",
|
||||||
|
"webworker"
|
||||||
|
],
|
||||||
|
"alwaysStrict": true,
|
||||||
|
"strict": true,
|
||||||
|
"preserveConstEnums": true,
|
||||||
|
"sourceMap": true,
|
||||||
|
"esModuleInterop": true,
|
||||||
|
"types": [
|
||||||
|
"@cloudflare/workers-types"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"include": [
|
||||||
|
"src/",
|
||||||
|
],
|
||||||
|
"exclude": [
|
||||||
|
"node_modules/",
|
||||||
|
"dist/",
|
||||||
|
]
|
||||||
|
}
|
10
workers-site/package.json
Normal file
10
workers-site/package.json
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
{
|
||||||
|
"private": true,
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "HTTP requests handler",
|
||||||
|
"main": "index.js",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@cloudflare/kv-asset-handler": "~0.1.2"
|
||||||
|
}
|
||||||
|
}
|
43
wrangler.toml
Normal file
43
wrangler.toml
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
# Default worker is for dev only.
|
||||||
|
# See omaps and organicmaps environments below for production.
|
||||||
|
name = 'url-processor'
|
||||||
|
type = 'javascript'
|
||||||
|
# Organic Maps CF Account ID.
|
||||||
|
account_id = '462f578f0939f041e2c24ec99adce458'
|
||||||
|
workers_dev = true
|
||||||
|
|
||||||
|
[site]
|
||||||
|
bucket = './public'
|
||||||
|
entry-point = './workers-site'
|
||||||
|
|
||||||
|
[build]
|
||||||
|
upload.format = 'service-worker'
|
||||||
|
command = 'npm i --prefer-offline --no-audit && npm run build'
|
||||||
|
|
||||||
|
[vars]
|
||||||
|
DEBUG = true
|
||||||
|
|
||||||
|
|
||||||
|
[env.omaps]
|
||||||
|
name = 'url-processor-omaps'
|
||||||
|
workers_dev = false
|
||||||
|
# omaps.app CF zone ID.
|
||||||
|
zone_id = '3fce06554abc3899504e11d928be0ee7'
|
||||||
|
# See the full list of handled files in the code.
|
||||||
|
route = 'omaps.app/*'
|
||||||
|
|
||||||
|
[env.omaps.vars]
|
||||||
|
DEBUG = false
|
||||||
|
|
||||||
|
|
||||||
|
#[env.organicmaps]
|
||||||
|
#DEBUG = false
|
||||||
|
#name = 'url-processor-organicmaps'
|
||||||
|
#workers_dev = false
|
||||||
|
# organicmaps.app CF zone ID.
|
||||||
|
#zone_id = 'a520ad91909e819d66c62f53f9454589'
|
||||||
|
# See the full list of handled files in the code.
|
||||||
|
#route = 'organicmaps.app/*'
|
||||||
|
|
||||||
|
[env.organicmaps.vars]
|
||||||
|
DEBUG = false
|
Loading…
Add table
Reference in a new issue