You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
102 lines
3.0 KiB
TypeScript
102 lines
3.0 KiB
TypeScript
import { restClient } from "@polygon.io/client-js";
|
|
import { client } from "./clickhouse.js";
|
|
import PQueue from "p-queue";
|
|
import dotenv from "dotenv";
|
|
import { Env } from "@humanwhocodes/env";
|
|
|
|
dotenv.config();
|
|
|
|
const env = new Env();
|
|
const POLYGON_API_KEY = env.require("POLYGON_API_KEY");
|
|
const CONCURRENCY = Number.parseInt(env.get("CONCURRENCY", "6"));
|
|
|
|
const polygon = restClient(POLYGON_API_KEY, "https://api.polygon.io", {
|
|
pagination: true,
|
|
});
|
|
|
|
const queue = new PQueue({ concurrency: CONCURRENCY });
|
|
|
|
function* dateRange(
|
|
startDateStr: string,
|
|
endDateStr: string
|
|
): Generator<string, void, unknown> {
|
|
// Convert the start and end date strings to Date objects
|
|
const startDate = new Date(startDateStr);
|
|
const endDate = new Date(endDateStr);
|
|
|
|
// Loop from the start date to the end date
|
|
for (
|
|
let currentDate = startDate;
|
|
currentDate <= endDate;
|
|
currentDate.setDate(currentDate.getDate() + 1)
|
|
) {
|
|
// Format the current date as YYYY-MM-DD
|
|
const formattedDate = currentDate.toISOString().split("T")[0];
|
|
|
|
// Yield the formatted date
|
|
yield formattedDate;
|
|
}
|
|
}
|
|
|
|
for (const date of dateRange("2024-05-02", "2024-05-03")) {
|
|
const contracts = await polygon.reference.optionsContracts({
|
|
underlying_ticker: "SPY",
|
|
as_of: date,
|
|
limit: 1000,
|
|
});
|
|
if (contracts.status?.toLowerCase() === "ok" && contracts.results) {
|
|
for (const contract of contracts.results) {
|
|
/** I don't know why we need this `if` again, but typescript yelled at me: */
|
|
if (contract.ticker) {
|
|
await queue.add(async () => {
|
|
if (contract.ticker) {
|
|
const quotes = await polygon.options.quotes(
|
|
contract.ticker,
|
|
{
|
|
timestamp: date,
|
|
sort: "timestamp",
|
|
order: "asc",
|
|
limit: 50000,
|
|
},
|
|
{ pagination: true }
|
|
);
|
|
if (quotes.status?.toLowerCase() === "ok" && quotes.results) {
|
|
await client.insert({
|
|
table: "option_quotes2",
|
|
values: quotes.results.map(
|
|
({ bid_price, ask_price, sip_timestamp }) => ({
|
|
option_type: contract.contract_type,
|
|
symbol: contract.underlying_ticker,
|
|
expiration_date: contract.expiration_date,
|
|
strike: contract.strike_price,
|
|
ts: new Date(Math.floor(sip_timestamp / 1000)),
|
|
bid: bid_price,
|
|
ask: ask_price,
|
|
})
|
|
),
|
|
format: "JSONEachRow",
|
|
});
|
|
console.log(date, contract.ticker, quotes.results?.length);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// const data = await polygon.options.quotes(
|
|
// "O:SPY241220P00600000",
|
|
// {
|
|
// "timestamp.gte": "2024-12-15",
|
|
// "timestamp.lte": "2024-12-16",
|
|
// sort: "timestamp",
|
|
// order: "asc",
|
|
// limit: 50000,
|
|
// },
|
|
// { pagination: true }
|
|
// );
|
|
|
|
// console.log(data.status);
|
|
// console.log(data.results?.length);
|