Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

improve: Add external config #1362

Open
wants to merge 62 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
62 commits
Select commit Hold shift + click to select a range
a68d5e7
improve: Add external config
evaldofelipe Jan 8, 2025
b0adb9d
add test route
evaldofelipe Jan 8, 2025
0ea8765
update env
evaldofelipe Jan 8, 2025
76ceda7
test
evaldofelipe Jan 8, 2025
366fdec
test
evaldofelipe Jan 8, 2025
c6bc7a4
update env
evaldofelipe Jan 8, 2025
a4a28ff
add pre-build env
evaldofelipe Jan 8, 2025
b231bc1
test
evaldofelipe Jan 8, 2025
7b6b27b
test
evaldofelipe Jan 8, 2025
5a42a29
update
evaldofelipe Jan 8, 2025
13fc109
update test
evaldofelipe Jan 9, 2025
8b93e40
fix
evaldofelipe Jan 9, 2025
7d754e0
test
evaldofelipe Jan 9, 2025
937b85c
update
evaldofelipe Jan 9, 2025
7e2e8bc
test
evaldofelipe Jan 9, 2025
4a26327
test
evaldofelipe Jan 9, 2025
e12c12d
change path
evaldofelipe Jan 9, 2025
51e3dcb
update test
evaldofelipe Jan 9, 2025
0f831e2
fix export
evaldofelipe Jan 9, 2025
9bb75a8
update
evaldofelipe Jan 9, 2025
8c90d4c
update
evaldofelipe Jan 10, 2025
851eb04
test
evaldofelipe Jan 10, 2025
6c3c929
test
evaldofelipe Jan 10, 2025
e5686ea
test tmp
evaldofelipe Jan 10, 2025
c1d6c20
udpate
evaldofelipe Jan 10, 2025
387e441
test
evaldofelipe Jan 10, 2025
ad5591f
test
evaldofelipe Jan 10, 2025
e480a7c
test
evaldofelipe Jan 10, 2025
919e9b8
test
evaldofelipe Jan 10, 2025
51529fe
test
evaldofelipe Jan 10, 2025
6bba313
test
evaldofelipe Jan 10, 2025
d704522
testt
evaldofelipe Jan 10, 2025
fc0cfdf
update
evaldofelipe Jan 10, 2025
4928c36
update
evaldofelipe Jan 10, 2025
085bc05
test
evaldofelipe Jan 10, 2025
cc57876
test
evaldofelipe Jan 10, 2025
47ac5b1
test
evaldofelipe Jan 10, 2025
fc07c1f
test
evaldofelipe Jan 10, 2025
13b235e
update
evaldofelipe Jan 10, 2025
c081d02
update test
evaldofelipe Jan 10, 2025
3207e3e
test
evaldofelipe Jan 10, 2025
8715c0f
test
evaldofelipe Jan 13, 2025
2293c1d
test
evaldofelipe Jan 14, 2025
ab82534
update
evaldofelipe Jan 14, 2025
6e04148
test
evaldofelipe Jan 14, 2025
6080af6
test
evaldofelipe Jan 14, 2025
2c1014c
chore: you're welcome
james-a-morris Jan 15, 2025
ec70f38
test update
mrice32 Jan 15, 2025
b253998
test
mrice32 Jan 15, 2025
f7a6404
Update available-routes.ts
mrice32 Jan 15, 2025
22bd246
Update available-routes.ts
mrice32 Jan 15, 2025
653d944
read dotenv file manually
mrice32 Jan 15, 2025
f5a2092
test
evaldofelipe Jan 21, 2025
caf9b61
update test
evaldofelipe Jan 21, 2025
6b812a7
update
evaldofelipe Jan 21, 2025
e1c11a5
test var
evaldofelipe Jan 22, 2025
b64892b
WIP
mrice32 Jan 22, 2025
600560e
WIP
mrice32 Jan 22, 2025
c925e75
wip
evaldofelipe Jan 22, 2025
399880e
update
evaldofelipe Jan 22, 2025
80994e6
wip
evaldofelipe Jan 22, 2025
565bf16
Update env var method
evaldofelipe Jan 22, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -63,3 +63,8 @@ stats.*
/blob-report/
/playwright/.cache/
.cache-synpress

# ENV
output_api.env
output.env
output.log
4 changes: 3 additions & 1 deletion api/_cache.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { createClient, VercelKV } from "@vercel/kv";
import { interfaces } from "@across-protocol/sdk";
import { getEnvs } from "./_env";

const {
KV_REST_API_READ_ONLY_TOKEN,
Expand All @@ -8,7 +9,8 @@ const {
UPSTASH_REDIS_REST_URL,
UPSTASH_REDIS_REST_TOKEN,
UPSTASH_REDIS_READ_ONLY_TOKEN,
} = process.env;
} = getEnvs();

const isRedisCacheEnabled =
(KV_REST_API_URL && (KV_REST_API_TOKEN || KV_REST_API_READ_ONLY_TOKEN)) ||
(UPSTASH_REDIS_REST_URL &&
Expand Down
16 changes: 12 additions & 4 deletions api/_constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,14 @@ import { ethers } from "ethers";
import { relayFeeCalculator, utils } from "@across-protocol/sdk";
import * as constants from "@across-protocol/constants";

import { getEnvs } from "./_env";

const {
RELAYER_ADDRESS_OVERRIDES,
GRAPH_API_KEY,
RELAYER_FEE_CAPITAL_COST_OVERRIDES,
} = getEnvs();

export const CHAIN_IDs = constants.CHAIN_IDs;
export const TOKEN_SYMBOLS_MAP = constants.TOKEN_SYMBOLS_MAP;

Expand Down Expand Up @@ -122,15 +130,15 @@ export const defaultRelayerAddressOverride: {
chains?: { [chainId: string]: string };
};
};
} = JSON.parse(process.env.RELAYER_ADDRESS_OVERRIDES || "{}");
} = JSON.parse(RELAYER_ADDRESS_OVERRIDES || "{}");

export const graphAPIKey = process.env.GRAPH_API_KEY;
export const graphAPIKey = GRAPH_API_KEY;

const relayerFeeCapitalCostOverrides: Record<
string,
Record<string, Record<string, relayFeeCalculator.CapitalCostConfig>>
> = process.env.RELAYER_FEE_CAPITAL_COST_OVERRIDES
? JSON.parse(process.env.RELAYER_FEE_CAPITAL_COST_OVERRIDES)
> = RELAYER_FEE_CAPITAL_COST_OVERRIDES
? JSON.parse(RELAYER_FEE_CAPITAL_COST_OVERRIDES)
: {};

export const relayerFeeCapitalCostConfig: {
Expand Down
8 changes: 7 additions & 1 deletion api/_dexes/1inch.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,12 @@ import axios from "axios";

import { AcrossSwap, SwapQuoteAndCalldata } from "./types";
import { getSwapAndBridgeAddress } from "./utils";
import { getEnvs } from "../_env";

const {
ONEINCH_API_KEY,
} = getEnvs();


export async function get1inchQuoteAndCalldata(
swap: AcrossSwap
Expand All @@ -12,7 +18,7 @@ export async function get1inchQuoteAndCalldata(
);
const apiBaseUrl = `https://api.1inch.dev/swap/v6.0/${swap.swapToken.chainId}`;
const apiHeaders = {
Authorization: `Bearer ${process.env.ONEINCH_API_KEY}`,
Authorization: `Bearer ${ONEINCH_API_KEY}`,
accept: "application/json",
};

Expand Down
14 changes: 14 additions & 0 deletions api/_env.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import path from "path";
import fs from "fs";
import dotenv from "dotenv";

let envPath = path.join(process.cwd(), "output_vercel_api.env");
let envFile = fs.readFileSync(envPath, "utf-8");
dotenv.populate(
process.env as dotenv.DotenvPopulateInput,
dotenv.parse(envFile)
);

export const getEnvs = () => {
return process.env;
};
29 changes: 18 additions & 11 deletions api/_utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -75,11 +75,7 @@ import {
InvalidParamError,
RouteNotEnabledError,
} from "./_errors";

export { InputError, handleErrorCondition } from "./_errors";

type LoggingUtility = sdk.relayFeeCalculator.Logger;
type RpcProviderName = keyof typeof rpcProvidersJson.providers.urls;
import { getEnvs } from "./_env";

const {
REACT_APP_HUBPOOL_CHAINID,
Expand All @@ -89,7 +85,18 @@ const {
PRIORITY_FEE_MARKUP,
VERCEL_ENV,
LOG_LEVEL,
} = process.env;
REACT_APP_DISABLED_CHAINS,
REACT_APP_DISABLED_CHAINS_FOR_AVAILABLE_ROUTES,
REACT_APP_DISABLED_TOKENS_FOR_AVAILABLE_ROUTES,
VERCEL_URL,
} = getEnvs();


export { InputError, handleErrorCondition } from "./_errors";

type LoggingUtility = sdk.relayFeeCalculator.Logger;
type RpcProviderName = keyof typeof rpcProvidersJson.providers.urls;


export const baseFeeMarkup: {
[chainId: string]: number;
Expand All @@ -115,19 +122,19 @@ export const DISABLED_ROUTE_TOKENS = (
// temporarily without having to redeploy the app or change core config
// data (e.g. the ENABLED_ROUTES object and the data/routes.json files).
export const DISABLED_CHAINS = (
process.env.REACT_APP_DISABLED_CHAINS || ""
REACT_APP_DISABLED_CHAINS || ""
).split(",");

// This is an array of chainIds that should be disabled. In contrast to the
// above constant `DISABLED_CHAINS`, this constant is used to disable chains
// only for the `/available-routes` endpoint and DOES NOT affect the
// `ENABLED_ROUTES` object.
export const DISABLED_CHAINS_FOR_AVAILABLE_ROUTES = (
process.env.REACT_APP_DISABLED_CHAINS_FOR_AVAILABLE_ROUTES || ""
REACT_APP_DISABLED_CHAINS_FOR_AVAILABLE_ROUTES || ""
).split(",");

export const DISABLED_TOKENS_FOR_AVAILABLE_ROUTES = (
process.env.REACT_APP_DISABLED_TOKENS_FOR_AVAILABLE_ROUTES || ""
REACT_APP_DISABLED_TOKENS_FOR_AVAILABLE_ROUTES || ""
).split(",");

const _ENABLED_ROUTES =
Expand Down Expand Up @@ -196,8 +203,8 @@ export const getLogger = (): LoggingUtility => {
* @returns A valid URL of the current endpoint in vercel
*/
export const resolveVercelEndpoint = () => {
const url = process.env.VERCEL_URL ?? "across.to";
const env = process.env.VERCEL_ENV ?? "development";
const url = VERCEL_URL ?? "across.to";
const env = VERCEL_ENV ?? "development";
switch (env) {
case "preview":
case "production":
Expand Down
124 changes: 12 additions & 112 deletions api/available-routes.ts
Original file line number Diff line number Diff line change
@@ -1,16 +1,13 @@
import { VercelResponse } from "@vercel/node";
import { object, assert, Infer, optional, string } from "superstruct";
import {
getLogger,
applyMapFilter,
validAddress,
positiveIntStr,
ENABLED_ROUTES,
handleErrorCondition,
DISABLED_CHAINS_FOR_AVAILABLE_ROUTES,
DISABLED_TOKENS_FOR_AVAILABLE_ROUTES,
} from "./_utils";
import { object, Infer, optional, string } from "superstruct";
import dotenv from "dotenv";
import { validAddress, positiveIntStr } from "./_utils";
import { TypedVercelRequest } from "./_types";
import { getEnvs } from "./_env";

const {
GIT_ENV_EXPORTED,
} = getEnvs();

const AvailableRoutesQueryParamsSchema = object({
originToken: optional(validAddress()),
Expand All @@ -25,108 +22,11 @@ type AvailableRoutesQueryParams = Infer<
typeof AvailableRoutesQueryParamsSchema
>;

const handler = async (
{ query }: TypedVercelRequest<AvailableRoutesQueryParams>,
response: VercelResponse
) => {
const logger = getLogger();
logger.debug({
at: "Routes",
message: "Query data",
query,
const handler = async (_: any, response: VercelResponse) => {
response.status(200).json({
test: GIT_ENV_EXPORTED,
});
try {
assert(query, AvailableRoutesQueryParamsSchema);

const {
originToken,
destinationToken,
originChainId,
destinationChainId,
originTokenSymbol,
destinationTokenSymbol,
} = query;

const enabledRoutes = applyMapFilter(
ENABLED_ROUTES.routes,
// Filter out elements from the request query parameters
(route: {
originToken: string;
originChainId: number;
destinationChainId: number;
destinationToken: string;
fromTokenSymbol: string;
toTokenSymbol: string;
isNative: boolean;
}) =>
![route.originChainId, route.destinationChainId].some((chainId) =>
DISABLED_CHAINS_FOR_AVAILABLE_ROUTES.includes(String(chainId))
) &&
!DISABLED_TOKENS_FOR_AVAILABLE_ROUTES.some(
(s) => s.toUpperCase() === route.fromTokenSymbol.toUpperCase()
) &&
(!originToken ||
originToken.toLowerCase() === route.originToken.toLowerCase()) &&
(!originChainId || originChainId === String(route.originChainId)) &&
(!destinationChainId ||
destinationChainId === String(route.destinationChainId)) &&
(!destinationToken ||
destinationToken.toLowerCase() ===
route.destinationToken.toLowerCase()) &&
(!originTokenSymbol ||
originTokenSymbol.toUpperCase() ===
route.fromTokenSymbol.toUpperCase()) &&
(!destinationTokenSymbol ||
destinationTokenSymbol.toUpperCase() ===
route.toTokenSymbol.toUpperCase()),
// Create a mapping of enabled routes to a route with the destination token resolved.
(route) => ({
originChainId: route.fromChain,
originToken: route.fromTokenAddress,
destinationChainId: route.toChain,
fromTokenSymbol: route.fromTokenSymbol,
toTokenSymbol: route.toTokenSymbol,
destinationToken: route.toTokenAddress,
isNative: route.isNative,
})
).map((route) => ({
originChainId: route.originChainId,
originToken: route.originToken,
destinationChainId: route.destinationChainId,
destinationToken: route.destinationToken,
originTokenSymbol: route.fromTokenSymbol,
destinationTokenSymbol: route.toTokenSymbol,
isNative: route.isNative,
}));

// Two different explanations for how `stale-while-revalidate` works:

// https://vercel.com/docs/concepts/edge-network/caching#stale-while-revalidate
// This tells our CDN the value is fresh for 6 hours. If a request is repeated within the next 6 hours,
// the previously cached value is still fresh. The header x-vercel-cache present in the response will show the
// value HIT. If the request is repeated up to 6 hours later, the cached value will be stale but
// still render. In the background, a revalidation request will be made to populate the cache with a fresh value.
// x-vercel-cache will have the value STALE until the cache is refreshed.

// https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Cache-Control
// The response is fresh for 6 hours. After 6 hours it becomes stale, but the cache is allowed to reuse it
// for any requests that are made in the following 6 hours, provided that they revalidate the response in the background.
// Revalidation will make the cache be fresh again, so it appears to clients that it was always fresh during
// that period — effectively hiding the latency penalty of revalidation from them.
// If no request happened during that period, the cache became stale and the next request will revalidate normally.
logger.debug({
at: "Routes",
message: "Response data",
responseJson: enabledRoutes,
});
response.setHeader(
"Cache-Control",
"s-maxage=21600, stale-while-revalidate=21600"
);
response.status(200).json(enabledRoutes);
} catch (error: unknown) {
return handleErrorCondition("available-routes", response, logger, error);
}
return;
};

export default handler;
4 changes: 3 additions & 1 deletion api/coingecko.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,11 +24,13 @@ import { InvalidParamError } from "./_errors";
import { coingecko } from "@across-protocol/sdk";

const { Coingecko } = coingecko;

import { getEnvs } from "./_env";
const {
REACT_APP_COINGECKO_PRO_API_KEY,
REDIRECTED_TOKEN_PRICE_LOOKUP_ADDRESSES,
BALANCER_V2_TOKENS,
} = process.env;
} = getEnvs();

const CoingeckoQueryParamsSchema = object({
l1Token: optional(validAddress()),
Expand Down
4 changes: 3 additions & 1 deletion api/cron-cache-balances.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,8 @@ import { UnauthorizedError } from "./_errors";

import mainnetChains from "../src/data/chains_1.json";

import { getEnvs } from "./_env";

const handler = async (
request: TypedVercelRequest<Record<string, never>>,
response: VercelResponse
Expand All @@ -34,7 +36,7 @@ const handler = async (
const {
REACT_APP_FULL_RELAYERS, // These are relayers running a full auto-rebalancing strategy.
REACT_APP_TRANSFER_RESTRICTED_RELAYERS, // These are relayers whose funds stay put.
} = process.env;
} = getEnvs();

const fullRelayers = !REACT_APP_FULL_RELAYERS
? []
Expand Down
3 changes: 2 additions & 1 deletion api/limits.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import { BigNumber, ethers } from "ethers";
import { CHAIN_IDs, DEFAULT_SIMULATED_RECIPIENT_ADDRESS } from "./_constants";
import { TokenInfo, TypedVercelRequest } from "./_types";
import { object, assert, Infer, optional, string } from "superstruct";
import { getEnvs } from "./_env";

import {
ENABLED_ROUTES,
Expand Down Expand Up @@ -67,7 +68,7 @@ const handler = async (
REACT_APP_TRANSFER_RESTRICTED_RELAYERS, // These are relayers whose funds stay put.
MIN_DEPOSIT_USD, // The global minimum deposit in USD for all destination chains. The minimum deposit
// returned by the relayerFeeDetails() call will be floor'd with this value (after converting to token units).
} = process.env;
} = getEnvs();
const provider = getProvider(HUB_POOL_CHAIN_ID);

const fullRelayers = !REACT_APP_FULL_RELAYERS
Expand Down
3 changes: 2 additions & 1 deletion api/suggested-fees.ts
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ import {
AmountTooHighError,
AmountTooLowError,
} from "./_errors";
import { getEnvs } from "./_env";

const { BigNumber } = ethers;

Expand Down Expand Up @@ -70,7 +71,7 @@ const handler = async (
query,
});
try {
const { QUOTE_BLOCK_BUFFER, QUOTE_BLOCK_PRECISION } = process.env;
const { QUOTE_BLOCK_BUFFER, QUOTE_BLOCK_PRECISION } = getEnvs();

const provider = getProvider(HUB_POOL_CHAIN_ID, {
useSpeedProvider: true,
Expand Down
3 changes: 2 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -54,8 +54,9 @@
"scripts": {
"start": "export REACT_APP_GIT_COMMIT_HASH=$(git rev-parse HEAD) && vite",
"dev": "export REACT_APP_GIT_COMMIT_HASH=$(git rev-parse HEAD) && vite --port $PORT --host",
"build": "yarn prebuild && export REACT_APP_GIT_COMMIT_HASH=$(git rev-parse HEAD) && tsc && vite build",
"build": "yarn prebuild-env && yarn prebuild && . ./output_vercel_build.env && echo $GIT_ENV_EXPORTED && export REACT_APP_GIT_COMMIT_HASH=$(git rev-parse HEAD) && tsc && vite build",
"prebuild": "tsx scripts/pre-build.ts",
"prebuild-env": "./scripts/pre-build-env.sh",
"analyze": "yarn build && rollup-plugin-visualizer --open ./bundle-size-analysis.json",
"test": "export REACT_APP_GIT_COMMIT_HASH=$(git rev-parse HEAD) && jest --env jsdom src",
"serve": "vite preview --port 3000",
Expand Down
Loading
Loading