@okx-dex/okx-dex-sdk
Version:
OKX DEX SDK
118 lines (117 loc) • 4.85 kB
JavaScript
;
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.HTTPClient = void 0;
const crypto_js_1 = __importDefault(require("crypto-js"));
class APIError extends Error {
constructor(message, status, statusText, responseBody, requestDetails) {
super(message);
this.status = status;
this.statusText = statusText;
this.responseBody = responseBody;
this.requestDetails = requestDetails;
this.name = 'APIError';
}
}
class HTTPClient {
constructor(config) {
this.config = {
baseUrl: 'https://web3.okx.com',
maxRetries: 3,
timeout: 30000,
...config
};
}
getHeaders(timestamp, method, path, queryString = "") {
const stringToSign = timestamp + method + path + queryString;
// Ensure the string is properly encoded
const encodedString = crypto_js_1.default.enc.Utf8.parse(stringToSign);
const secretKey = crypto_js_1.default.enc.Utf8.parse(this.config.secretKey);
// Create HMAC-SHA256 signature
const signature = crypto_js_1.default.HmacSHA256(encodedString, secretKey);
return {
"Content-Type": "application/json",
"OK-ACCESS-KEY": this.config.apiKey,
"OK-ACCESS-SIGN": crypto_js_1.default.enc.Base64.stringify(signature),
"OK-ACCESS-TIMESTAMP": timestamp,
"OK-ACCESS-PASSPHRASE": this.config.apiPassphrase,
"OK-ACCESS-PROJECT": this.config.projectId,
};
}
async handleErrorResponse(response, requestDetails) {
let responseBody;
try {
responseBody = await response.json();
}
catch (e) {
responseBody = await response.text();
}
throw new APIError(`HTTP error! status: ${response.status}`, response.status, response.statusText, responseBody, requestDetails);
}
async request(method, path, params) {
const timestamp = new Date().toISOString();
// Filter out undefined values from params
const cleanParams = params ? Object.fromEntries(Object.entries(params).filter(([_, v]) => v !== undefined)) : undefined;
const queryString = cleanParams ? "?" + new URLSearchParams(cleanParams).toString() : "";
const headers = this.getHeaders(timestamp, method, path, queryString);
const requestDetails = {
method,
path,
params: cleanParams,
queryString,
url: `${this.config.baseUrl}${path}${queryString}`
};
// Log request details in development
if (process.env.NODE_ENV === 'development') {
console.log('Request Details:', {
url: requestDetails.url,
method: requestDetails.method,
headers: {
...headers,
'OK-ACCESS-SIGN': '***', // Hide sensitive data
'OK-ACCESS-KEY': '***',
'OK-ACCESS-PASSPHRASE': '***'
},
params: requestDetails.params
});
}
let retries = 0;
while (retries < this.config.maxRetries) {
try {
const response = await fetch(`${this.config.baseUrl}${path}${queryString}`, {
method,
headers
});
if (!response.ok) {
await this.handleErrorResponse(response, requestDetails);
}
const data = await response.json();
// Log response in development
if (process.env.NODE_ENV === 'development') {
console.log('Response:', JSON.stringify(data, null, 2));
}
if (data.code !== "0") {
throw new APIError(`API Error: ${data.msg}`, response.status, response.statusText, data, requestDetails);
}
return data;
}
catch (error) {
if (error instanceof APIError) {
if (retries === this.config.maxRetries - 1)
throw error;
}
else {
if (retries === this.config.maxRetries - 1) {
throw new APIError(error instanceof Error ? error.message : 'Unknown error', undefined, undefined, undefined, requestDetails);
}
}
retries++;
await new Promise(resolve => setTimeout(resolve, 1000 * retries));
}
}
throw new Error("Max retries exceeded");
}
}
exports.HTTPClient = HTTPClient;