From 223238f33f2745bf7e9ce33d858dc0f779e4e5c8 Mon Sep 17 00:00:00 2001 From: Aponia Date: Sun, 15 Oct 2023 20:58:29 -0700 Subject: [PATCH] feat: proxy for mapbox (#730) Co-authored-by: Eric Pedley --- .github/workflows/step_setup_and_build.yaml | 1 + .github/workflows/test.yaml | 1 + apps/antalmanac/src/components/Map/Map.tsx | 13 ++-- apps/antalmanac/src/components/Map/Routes.tsx | 7 +- apps/antalmanac/src/lib/api/endpoints.ts | 4 +- apps/backend/.env.sample | 3 + apps/backend/src/env.ts | 1 + apps/backend/src/index.ts | 25 ++++++- apps/cdk/.env.sample | 3 +- apps/cdk/lib/app.ts | 38 ++++------ apps/cdk/lib/backend.ts | 71 ++++++++----------- 11 files changed, 91 insertions(+), 76 deletions(-) diff --git a/.github/workflows/step_setup_and_build.yaml b/.github/workflows/step_setup_and_build.yaml index a584c741f..ebc336bd7 100644 --- a/.github/workflows/step_setup_and_build.yaml +++ b/.github/workflows/step_setup_and_build.yaml @@ -64,6 +64,7 @@ jobs: HOSTED_ZONE_ID: ${{ secrets.HOSTED_ZONE_ID }} CERTIFICATE_ARN: ${{ secrets.CERTIFICATE_ARN }} MONGODB_URI_PROD: ${{ secrets.MONGODB_URI_PROD }} + MAPBOX_ACCESS_TOKEN: ${{ secrets.MAPBOX_ACCESS_TOKEN }} PR_NUM: ${{ github.event.pull_request.number }} # cdk env variables API_SUB_DOMAIN: ${{ env.apiSubDomain }} diff --git a/.github/workflows/test.yaml b/.github/workflows/test.yaml index d72a3a765..a70b7b362 100644 --- a/.github/workflows/test.yaml +++ b/.github/workflows/test.yaml @@ -51,6 +51,7 @@ jobs: HOSTED_ZONE_ID: ${{ secrets.HOSTED_ZONE_ID }} CERTIFICATE_ARN: ${{ secrets.CERTIFICATE_ARN }} MONGODB_URI_PROD: ${{ secrets.MONGODB_URI_PROD }} + MAPBOX_ACCESS_TOKEN: ${{ secrets.MAPBOX_ACCESS_TOKEN }} PR_NUM: ${{ github.event.pull_request.number }} # CDK environment variables. diff --git a/apps/antalmanac/src/components/Map/Map.tsx b/apps/antalmanac/src/components/Map/Map.tsx index 2f33e5715..c59c91724 100644 --- a/apps/antalmanac/src/components/Map/Map.tsx +++ b/apps/antalmanac/src/components/Map/Map.tsx @@ -14,14 +14,11 @@ import locationIds from '$lib/location_ids'; import buildingCatalogue from '$lib/buildingCatalogue'; import type { Building } from '$lib/buildingCatalogue'; import type { CourseEvent } from '$components/Calendar/CourseCalendarEvent'; - -const ACCESS_TOKEN = 'pk.eyJ1IjoicGVkcmljIiwiYSI6ImNsZzE0bjk2ajB0NHEzanExZGFlbGpwazIifQ.l14rgv5vmu5wIMgOUUhUXw'; +import { MAPBOX_PROXY_TILES_ENDPOINT } from '$lib/api/endpoints'; const ATTRIBUTION_MARKUP = '© OpenStreetMap contributors | Images from UCI Map'; -const url = `https://api.mapbox.com/styles/v1/mapbox/streets-v11/tiles/{z}/{x}/{y}?access_token=${ACCESS_TOKEN}`; - const WORK_WEEK = ['All', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri']; const FULL_WEEK = ['All', 'Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat']; const weekendIndices = [0, 6]; @@ -231,7 +228,13 @@ export default function CourseMap() { /> - + diff --git a/apps/antalmanac/src/components/Map/Routes.tsx b/apps/antalmanac/src/components/Map/Routes.tsx index 81ee659cd..597e15f83 100644 --- a/apps/antalmanac/src/components/Map/Routes.tsx +++ b/apps/antalmanac/src/components/Map/Routes.tsx @@ -4,8 +4,7 @@ import type { LatLngTuple } from 'leaflet'; import 'leaflet-routing-machine'; import { createElementHook, createElementObject, useLeafletContext } from '@react-leaflet/core'; import type { LeafletContextInterface } from '@react-leaflet/core'; - -const ACCESS_TOKEN = 'pk.eyJ1IjoicGVkcmljIiwiYSI6ImNsZzE0bjk2ajB0NHEzanExZGFlbGpwazIifQ.l14rgv5vmu5wIMgOUUhUXw'; +import { MAPBOX_PROXY_DIRECTIONS_ENDPOINT } from '$lib/api/endpoints'; interface ClassRoutesProps { /** @@ -40,12 +39,14 @@ function createRouter(props: ClassRoutesProps, context: LeafletContextInterface) const waypoints = latLngTuples.map((latLngTuple) => L.latLng(latLngTuple)); const routerControl = L.Routing.control({ - router: L.Routing.mapbox(ACCESS_TOKEN, { + router: L.Routing.mapbox('', { /** * Default is mapbox/driving. More options: * @see {@link https://docs.mapbox.com/api/navigation/directions/#routing-profiles} */ profile: 'mapbox/walking', + + serviceUrl: MAPBOX_PROXY_DIRECTIONS_ENDPOINT, }), /** diff --git a/apps/antalmanac/src/lib/api/endpoints.ts b/apps/antalmanac/src/lib/api/endpoints.ts index 3cb2298e0..673b2cae8 100644 --- a/apps/antalmanac/src/lib/api/endpoints.ts +++ b/apps/antalmanac/src/lib/api/endpoints.ts @@ -4,7 +4,7 @@ function endpointTransform(path: string) { return `https://${import.meta.env.VITE_ENDPOINT}.api.antalmanac.com${path}`; } if (import.meta.env.VITE_LOCAL_SERVER) { - return `http://localhost:8080${path}`; + return `http://localhost:3000${path}`; } return import.meta.env.MODE === 'development' ? `https://dev.api.antalmanac.com${path}` @@ -13,6 +13,8 @@ function endpointTransform(path: string) { export const LOOKUP_NOTIFICATIONS_ENDPOINT = endpointTransform('/api/notifications/lookupNotifications'); export const REGISTER_NOTIFICATIONS_ENDPOINT = endpointTransform('/api/notifications/registerNotifications'); +export const MAPBOX_PROXY_DIRECTIONS_ENDPOINT = endpointTransform('/mapbox/directions'); +export const MAPBOX_PROXY_TILES_ENDPOINT = endpointTransform('/mapbox/tiles'); // PeterPortal API export const PETERPORTAL_GRAPHQL_ENDPOINT = 'https://api-next.peterportal.org/v1/graphql'; diff --git a/apps/backend/.env.sample b/apps/backend/.env.sample index f76d05f0d..c935e3941 100644 --- a/apps/backend/.env.sample +++ b/apps/backend/.env.sample @@ -9,3 +9,6 @@ USERDATA_TABLE_NAME=tablename # Provided by Lambda when running on AWS. AWS_REGION=us-east-1 + +# For Mapbox API +MAPBOX_ACCESS_TOKEN=pk.abc123 \ No newline at end of file diff --git a/apps/backend/src/env.ts b/apps/backend/src/env.ts index 53ba01944..a1e7db140 100644 --- a/apps/backend/src/env.ts +++ b/apps/backend/src/env.ts @@ -7,6 +7,7 @@ const Environment = type([ USERDATA_TABLE_NAME: 'string', AA_MONGODB_URI: 'string', AWS_REGION: 'string', + MAPBOX_ACCESS_TOKEN: 'string', 'PR_NUM?': 'number', }, '|>', diff --git a/apps/backend/src/index.ts b/apps/backend/src/index.ts index be9483e82..d87c27492 100644 --- a/apps/backend/src/index.ts +++ b/apps/backend/src/index.ts @@ -4,13 +4,15 @@ import type { CorsOptions } from 'cors'; import { createExpressMiddleware } from '@trpc/server/adapters/express'; import AppRouter from './routers'; import createContext from './context'; -import connectToMongoDB from '$db/mongodb'; import env from './env'; +import connectToMongoDB from '$db/mongodb'; const corsOptions: CorsOptions = { origin: ['https://antalmanac.com', 'https://www.antalmanac.com', 'https://icssc-projects.github.io/AntAlmanac'], }; +const MAPBOX_API_URL = 'https://api.mapbox.com'; + const PORT = 3000; export async function start(corsEnabled = false) { @@ -20,6 +22,27 @@ export async function start(corsEnabled = false) { app.use(cors(corsEnabled ? corsOptions : undefined)); app.use(express.json()); + app.use('/mapbox/directions/*', async (req, res) => { + const searchParams = new URLSearchParams(req.query as any); + searchParams.set('access_token', env.MAPBOX_ACCESS_TOKEN); + const url = `${MAPBOX_API_URL}/directions/v5/${req.params[0]}?${searchParams.toString()}`; + const result = await fetch(url).then((res) => res.text()); + res.send(result); + }); + + app.use('/mapbox/tiles/*', async (req, res) => { + const searchParams = new URLSearchParams(req.query as any); + searchParams.set('access_token', env.MAPBOX_ACCESS_TOKEN); + const url = `${MAPBOX_API_URL}/styles/v1/mapbox/streets-v11/tiles/${req.params[0]}?${searchParams.toString()}`; + const buffer = await fetch(url).then((res) => res.arrayBuffer()); + res.type('image/png') + res.send(Buffer.from(buffer)) + // // res.header('Content-Security-Policy', "img-src 'self'"); // https://stackoverflow.com/questions/56386307/loading-of-a-resource-blocked-by-content-security-policy + // // res.header('Access-Control-Allow-Methods', 'GET, OPTIONS') + // res.type('image/png') + // res.send(result) + }); + app.use( '/trpc', createExpressMiddleware({ diff --git a/apps/cdk/.env.sample b/apps/cdk/.env.sample index 5983407c2..7ef2814d6 100644 --- a/apps/cdk/.env.sample +++ b/apps/cdk/.env.sample @@ -1,2 +1,3 @@ MONGODB_URI_DEV=dev uri -MONGODB_URI_PROD=prod uri \ No newline at end of file +MONGODB_URI_PROD=prod uri +MAPBOX_ACCESS_TOKEN=pk.abc123 \ No newline at end of file diff --git a/apps/cdk/lib/app.ts b/apps/cdk/lib/app.ts index 1e345a641..65897253a 100644 --- a/apps/cdk/lib/app.ts +++ b/apps/cdk/lib/app.ts @@ -6,41 +6,30 @@ import FrontendStack from './frontend' const app = new App({ autoSynth: true }) // Check environmental variables -if ( - !process.env.CERTIFICATE_ARN || - !process.env.HOSTED_ZONE_ID || - !process.env.MONGODB_URI_PROD -) { +if (!process.env.CERTIFICATE_ARN || !process.env.HOSTED_ZONE_ID || !process.env.MONGODB_URI_PROD) { throw new Error('Missing environmental variables') } // Deploy staging if (process.env.PR_NUM) { const env: Environment = { region: 'us-east-1' } - new FrontendStack( - app, - `antalmanac-frontend-staging-${process.env.PR_NUM}`, - { + new FrontendStack(app, `antalmanac-frontend-staging-${process.env.PR_NUM}`, { + env, + stage: 'staging', + certificateArn: process.env.CERTIFICATE_ARN, + hostedZoneId: process.env.HOSTED_ZONE_ID, + prNum: process.env.PR_NUM, + }) + if (process.env.apiSubDomain !== 'dev') { + new BackendStack(app, `antalmanac-backend-staging-${process.env.PR_NUM}`, { env, stage: 'staging', certificateArn: process.env.CERTIFICATE_ARN, hostedZoneId: process.env.HOSTED_ZONE_ID, + mongoDbUriProd: process.env.MONGODB_URI_PROD, + mapboxAccessToken: process.env.MAPBOX_ACCESS_TOKEN, prNum: process.env.PR_NUM, - }, - ) - if (process.env.apiSubDomain !== 'dev') { - new BackendStack( - app, - `antalmanac-backend-staging-${process.env.PR_NUM}`, - { - env, - stage: 'staging', - certificateArn: process.env.CERTIFICATE_ARN, - hostedZoneId: process.env.HOSTED_ZONE_ID, - mongoDbUriProd: process.env.MONGODB_URI_PROD, - prNum: process.env.PR_NUM, - }, - ) + }) } } @@ -60,6 +49,7 @@ else { certificateArn: process.env.CERTIFICATE_ARN, hostedZoneId: process.env.HOSTED_ZONE_ID, mongoDbUriProd: process.env.MONGODB_URI_PROD, + mapboxAccessToken: process.env.MAPBOX_ACCESS_TOKEN, }) // prod frontend is deployed on GitHub Pages if (stage !== 'prod') { diff --git a/apps/cdk/lib/backend.ts b/apps/cdk/lib/backend.ts index 39e019a45..e14e3e2cd 100644 --- a/apps/cdk/lib/backend.ts +++ b/apps/cdk/lib/backend.ts @@ -11,6 +11,7 @@ import { transformUrl } from './helpers' export interface BackendProps extends StackProps { stage: string mongoDbUriProd: string + mapboxAccessToken: string hostedZoneId: string certificateArn: string prNum?: string @@ -20,39 +21,32 @@ export default class BackendStack extends Stack { constructor(scope: Construct, id: string, props: BackendProps) { super(scope, id, props) - const userDataDDB = new dynamnodb.Table( - this, - `antalmanac-userdata-ddb-${props.stage}`, - { - partitionKey: { - name: 'id', - type: dynamnodb.AttributeType.STRING, - }, - billingMode: dynamnodb.BillingMode.PAY_PER_REQUEST, - removalPolicy: - props.stage === 'dev' || props.stage === 'prod' - ? RemovalPolicy.RETAIN - : RemovalPolicy.DESTROY, + const userDataDDB = new dynamnodb.Table(this, `antalmanac-userdata-ddb-${props.stage}`, { + partitionKey: { + name: 'id', + type: dynamnodb.AttributeType.STRING, }, - ) + billingMode: dynamnodb.BillingMode.PAY_PER_REQUEST, + removalPolicy: + props.stage === 'dev' || props.stage === 'prod' + ? RemovalPolicy.RETAIN + : RemovalPolicy.DESTROY, + }) - const api = new lambda.Function( - this, - `antalmanac-api-${props.stage}-lambda`, - { - runtime: lambda.Runtime.NODEJS_18_X, - code: lambda.Code.fromAsset('../backend/dist'), - handler: 'lambda.handler', - timeout: Duration.seconds(5), - memorySize: 256, - environment: { - // We don't need dev database because we will never write to it - AA_MONGODB_URI: props.mongoDbUriProd, - STAGE: props.stage, - USERDATA_TABLE_NAME: userDataDDB.tableName, - }, + const api = new lambda.Function(this, `antalmanac-api-${props.stage}-lambda`, { + runtime: lambda.Runtime.NODEJS_18_X, + code: lambda.Code.fromAsset('../backend/dist'), + handler: 'lambda.handler', + timeout: Duration.seconds(5), + memorySize: 256, + environment: { + // We don't need dev database because we will never write to it + AA_MONGODB_URI: props.mongoDbUriProd, + STAGE: props.stage, + USERDATA_TABLE_NAME: userDataDDB.tableName, + MAPBOX_ACCESS_TOKEN: props.mapboxAccessToken, }, - ) + }) userDataDDB.grantReadWriteData(api) @@ -79,19 +73,14 @@ export default class BackendStack extends Stack { ), endpointType: apigateway.EndpointType.EDGE, }, + binaryMediaTypes: ['image/*'], }, ) - new route53.ARecord( - this, - `antalmanac-backend-a-record-${props.stage}`, - { - zone: zone, - recordName: transformUrl('api', props), - target: route53.RecordTarget.fromAlias( - new targets.ApiGateway(apiGateway), - ), - }, - ) + new route53.ARecord(this, `antalmanac-backend-a-record-${props.stage}`, { + zone: zone, + recordName: transformUrl('api', props), + target: route53.RecordTarget.fromAlias(new targets.ApiGateway(apiGateway)), + }) } }