Add PlacePage donations
https://docs.google.com/document/d/1f2zjYBr24Wlh1vBcTtCmisx-HidaYwivkXZGCHjlTTY/edit
This commit is contained in:
parent
05100af769
commit
14b002618e
10 changed files with 326 additions and 4 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -1,4 +1,5 @@
|
||||||
dist/
|
dist/
|
||||||
|
.wrangler/
|
||||||
# Autogenerated by esbuild.
|
# Autogenerated by esbuild.
|
||||||
workers-site/index.js
|
workers-site/index.js
|
||||||
node_modules/
|
node_modules/
|
||||||
|
|
|
@ -14,6 +14,10 @@ npm i
|
||||||
|
|
||||||
Use `npx wrangler dev` for localhost development and for testing using Cloudflare dev tools.
|
Use `npx wrangler dev` for localhost development and for testing using Cloudflare dev tools.
|
||||||
|
|
||||||
|
```
|
||||||
|
curl -H "X-OM-DataVersion: 241001" -H "X-OM-AppVersion: 2024.10.22-10-Google" -H 'Accept-Language: fr-FR' http://localhost:8787/maps
|
||||||
|
```
|
||||||
|
|
||||||
## Update node dependencies to their major versions
|
## Update node dependencies to their major versions
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
|
|
|
@ -7,8 +7,8 @@
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "esbuild src/index.ts --bundle --outfile=dist/index.js",
|
"build": "esbuild src/index.ts --bundle --outfile=dist/index.js",
|
||||||
"test": "jest",
|
"test": "jest",
|
||||||
"format": "prettier --write '{src,test}/**/*.{ts,tsx}' '*.json' '*.toml' '.github/**/*.yml'",
|
"format": "prettier --write '{src,test}/**/*.{ts,tsx,json}' '*.json' '*.toml' '.github/**/*.yml'",
|
||||||
"format:ci": "prettier --check '{src,test}/**/*.{ts,tsx}' '*.json' '*.toml' '.github/**/*.yml'",
|
"format:ci": "prettier --check '{src,test}/**/*.{ts,tsx,json}' '*.json' '*.toml' '.github/**/*.yml'",
|
||||||
"upgrade": "npx npm-check-updates -u && npm install",
|
"upgrade": "npx npm-check-updates -u && npm install",
|
||||||
"logs": "npx wrangler tail --env prod --format json"
|
"logs": "npx wrangler tail --env prod --format json"
|
||||||
},
|
},
|
||||||
|
|
44
src/locales.json
Normal file
44
src/locales.json
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
{
|
||||||
|
"en": {
|
||||||
|
"placePagePrompt": "Organic Maps app is free for everyone, thanks to your donations. No ads. No trackers. Open source.",
|
||||||
|
"perMonth": "/month",
|
||||||
|
"perYear": "/year",
|
||||||
|
"otherAmount": "Other"
|
||||||
|
},
|
||||||
|
"de": {
|
||||||
|
"placePagePrompt": "Organic Maps ist dank deiner Spenden für alle kostenlos. Keine Werbung. Keine Tracker. Open Source.",
|
||||||
|
"perMonth": "/Monat",
|
||||||
|
"perYear": "/Jahr",
|
||||||
|
"otherAmount": "Andere"
|
||||||
|
},
|
||||||
|
"fr": {
|
||||||
|
"placePagePrompt": "L'application Organic Maps est gratuite pour tout le monde grâce à vos dons. Pas de publicité. Pas de trackers. Open-source.",
|
||||||
|
"perMonth": "/mois",
|
||||||
|
"perYear": "/an",
|
||||||
|
"otherAmount": "Autre"
|
||||||
|
},
|
||||||
|
"nl": {
|
||||||
|
"placePagePrompt": "De Organic Maps app is gratis voor iedereen dankzij jullie donaties. Geen advertenties. Geen trackers. Open-source.",
|
||||||
|
"perMonth": "/maand",
|
||||||
|
"perYear": "/jaar",
|
||||||
|
"otherAmount": "Ander"
|
||||||
|
},
|
||||||
|
"it": {
|
||||||
|
"placePagePrompt": "L'app Organic Maps è gratuita per tutti grazie alle vostre donazioni. Nessuna pubblicità. Nessun tracker. Open-source.",
|
||||||
|
"perMonth": "/mese",
|
||||||
|
"perYear": "/anno",
|
||||||
|
"otherAmount": "Altro"
|
||||||
|
},
|
||||||
|
"es": {
|
||||||
|
"placePagePrompt": "Organic Maps es gratis para todos gracias a sus donaciones. Sin anuncios. Sin rastreadores. Código abierto.",
|
||||||
|
"perMonth": "/mes",
|
||||||
|
"perYear": "/año",
|
||||||
|
"otherAmount": "Otro"
|
||||||
|
},
|
||||||
|
"pt": {
|
||||||
|
"placePagePrompt": "O app Organic Maps é gratuito para todos graças às suas doações. Sem anúncios. Sem rastreadores. Código aberto.",
|
||||||
|
"perMonth": "/mês",
|
||||||
|
"perYear": "/ano",
|
||||||
|
"otherAmount": "Outro"
|
||||||
|
}
|
||||||
|
}
|
15
src/locales.ts
Normal file
15
src/locales.ts
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
import LOCALES_JSON from './locales.json';
|
||||||
|
|
||||||
|
export interface Locale {
|
||||||
|
placePagePrompt: string;
|
||||||
|
perMonth: string;
|
||||||
|
perYear: string;
|
||||||
|
otherAmount: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Locales {
|
||||||
|
[key: string]: Locale;
|
||||||
|
}
|
||||||
|
|
||||||
|
const LOCALES = LOCALES_JSON as Locales;
|
||||||
|
export default LOCALES;
|
198
src/products.json
Normal file
198
src/products.json
Normal file
|
@ -0,0 +1,198 @@
|
||||||
|
{
|
||||||
|
"FR": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"DE": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"NL": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"IT": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"ES": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"PT": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"BE": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"AT": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"LU": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"MC": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"AD": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"SM": [
|
||||||
|
{
|
||||||
|
"title": "4,99€$per_month",
|
||||||
|
"link": "https://donate.stripe.com/6oEg2912f2c81r200j"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "34,99€$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVa2bjdP19EA0mY6oI"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/28odU16mzg2Yc5GfYY"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"GB": [
|
||||||
|
{
|
||||||
|
"title": "£4.99$per_month",
|
||||||
|
"link": "https://donate.stripe.com/8wMg29fX98Awd9K28u"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "£34.99$per_year",
|
||||||
|
"link": "https://donate.stripe.com/eVabLT9yL2c89Xy7sP"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/6oE7vD26j3gc2v69AC"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"US": [
|
||||||
|
{
|
||||||
|
"title": "$5.49$per_month",
|
||||||
|
"link": "https://donate.stripe.com/00g3fncKXcQMedO5kL"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$36.99$per_year",
|
||||||
|
"link": "https://donate.stripe.com/bIY6rz5ivcQM9XyaF6"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "$other",
|
||||||
|
"link": "https://donate.stripe.com/7sIcPX5ivg2Y2v6145"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
40
src/products.ts
Normal file
40
src/products.ts
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
import LOCALES from './locales';
|
||||||
|
import PRODUCTS_JSON from './products.json';
|
||||||
|
|
||||||
|
export interface Product {
|
||||||
|
title: string;
|
||||||
|
link: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const PRODUCTS = PRODUCTS_JSON as Record<string, Product[]>;
|
||||||
|
|
||||||
|
export interface ProductsConfig {
|
||||||
|
placePagePrompt: string;
|
||||||
|
products: Product[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getProducts(locale: string | null): ProductsConfig | undefined {
|
||||||
|
if (!locale) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
const parts = locale.split(/[-_]/);
|
||||||
|
const language = parts[0].toLowerCase();
|
||||||
|
const country = parts[1] ? parts[1].toUpperCase() : '';
|
||||||
|
|
||||||
|
const products = PRODUCTS[country];
|
||||||
|
const trans = LOCALES[language];
|
||||||
|
if (products === undefined || trans === undefined) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
placePagePrompt: trans.placePagePrompt,
|
||||||
|
products: products.map((product) => ({
|
||||||
|
...product,
|
||||||
|
title: product.title
|
||||||
|
.replace('$other', trans.otherAmount)
|
||||||
|
.replace('$per_month', trans.perMonth)
|
||||||
|
.replace('$per_year', trans.perYear),
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
import { parseDataVersion, parseAppVersion } from './versions';
|
import { parseDataVersion, parseAppVersion } from './versions';
|
||||||
|
import { getProducts, ProductsConfig } from './products';
|
||||||
|
|
||||||
export const DATA_VERSIONS = [
|
export const DATA_VERSIONS = [
|
||||||
210529, //
|
210529, //
|
||||||
|
@ -125,7 +126,7 @@ export async function getServersList(request: Request) {
|
||||||
if (dataVersion === null) {
|
if (dataVersion === null) {
|
||||||
// Older clients download from the archive.
|
// Older clients download from the archive.
|
||||||
servers = [SERVER.backblaze];
|
servers = [SERVER.backblaze];
|
||||||
} else if (dataVersion == 240702 && abusedVersions.includes(request.headers.get('x-om-appversion'))) {
|
} else if (dataVersion == 240702 && abusedVersions.includes(request.headers.get('x-om-appversion') || 'unknown')) {
|
||||||
// Redirect https://apps.apple.com/us/app/mapxplorer-navigation-radar/id6463052823
|
// Redirect https://apps.apple.com/us/app/mapxplorer-navigation-radar/id6463052823
|
||||||
// who abuses our servers to a slow download "trap" node.
|
// who abuses our servers to a slow download "trap" node.
|
||||||
return new Response('["https://cdn-fi2.organicmaps.app/"]', {
|
return new Response('["https://cdn-fi2.organicmaps.app/"]', {
|
||||||
|
@ -176,6 +177,7 @@ export async function getServersList(request: Request) {
|
||||||
DonateUrl?: string;
|
DonateUrl?: string;
|
||||||
NY?: string;
|
NY?: string;
|
||||||
};
|
};
|
||||||
|
productsConfig?: ProductsConfig;
|
||||||
} = {
|
} = {
|
||||||
servers: servers,
|
servers: servers,
|
||||||
};
|
};
|
||||||
|
@ -203,11 +205,14 @@ export async function getServersList(request: Request) {
|
||||||
|
|
||||||
if (donatesEnabled) {
|
if (donatesEnabled) {
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
||||||
//@ts-ignore workarounds error TS2339: Property 'country' does not exist on type 'IncomingRequestCfProperties<unknown>'.
|
|
||||||
response.settings = {
|
response.settings = {
|
||||||
DonateUrl: DONATE_URL,
|
DonateUrl: DONATE_URL,
|
||||||
NY: 'false', // Must be `string` instead of `bool`, otherwise clients will crash
|
NY: 'false', // Must be `string` instead of `bool`, otherwise clients will crash
|
||||||
};
|
};
|
||||||
|
if (appVersion.code >= 241022) {
|
||||||
|
const locale = request.headers.get('accept-language');
|
||||||
|
response.productsConfig = getProducts(locale);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return new Response(JSON.stringify(response), {
|
return new Response(JSON.stringify(response), {
|
||||||
|
|
14
test/products.test.ts
Normal file
14
test/products.test.ts
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
import { describe, expect, test } from '@jest/globals';
|
||||||
|
import { getProducts } from '../src/products';
|
||||||
|
|
||||||
|
describe('getProducts', () => {
|
||||||
|
test('fr-FR', () => {
|
||||||
|
const fr_FR = getProducts('fr-FR');
|
||||||
|
expect(fr_FR).toBeDefined();
|
||||||
|
if (!fr_FR) return;
|
||||||
|
expect(fr_FR.placePagePrompt).toBe(
|
||||||
|
"L'application Organic Maps est gratuite pour tout le monde grâce à vos dons. Pas de publicité. Pas de trackers. Open-source.",
|
||||||
|
);
|
||||||
|
expect(fr_FR.products[fr_FR.products.length - 1].title).toEqual('Autre');
|
||||||
|
});
|
||||||
|
});
|
|
@ -11,6 +11,7 @@
|
||||||
"preserveConstEnums": true,
|
"preserveConstEnums": true,
|
||||||
"sourceMap": true,
|
"sourceMap": true,
|
||||||
"esModuleInterop": true,
|
"esModuleInterop": true,
|
||||||
|
"resolveJsonModule": true,
|
||||||
"types": ["@cloudflare/workers-types"]
|
"types": ["@cloudflare/workers-types"]
|
||||||
},
|
},
|
||||||
"include": ["src/"],
|
"include": ["src/"],
|
||||||
|
|
Loading…
Add table
Reference in a new issue