cc22174b78
- Implemented tests for fetching historical bars for AAPL with different timeframes (1D, 5Min, 1H). - Verified response structure and data integrity for each timeframe. - Ensured that the API returns valid data and appropriate status for the requests.
183 lines
7.7 KiB
TypeScript
183 lines
7.7 KiB
TypeScript
import { useLoaderData, useNavigate, useLocation } from "react-router";
|
|
import TradingViewChart from "../components/TradingViewChart";
|
|
import Navbar from "../components/Navbar";
|
|
|
|
export const meta = () => [{ title: "Stock Detail - AITrader" }];
|
|
|
|
interface LoaderData {
|
|
ticker: string;
|
|
position: number | null;
|
|
orders: any[];
|
|
bars: any[];
|
|
timeframe: string;
|
|
limit: number;
|
|
}
|
|
|
|
const TIMEFRAMES = [
|
|
{ value: "1D", label: "1 Day" },
|
|
{ value: "5Min", label: "5 Min" },
|
|
{ value: "15Min", label: "15 Min" },
|
|
{ value: "1H", label: "1 Hour" },
|
|
{ value: "1W", label: "1 Week" },
|
|
];
|
|
|
|
export async function loader({ params, request }: { params: { ticker: string }; request: Request }) {
|
|
const ticker = params.ticker?.toUpperCase() || "";
|
|
const url = new URL(request.url);
|
|
const timeframe = url.searchParams.get("timeframe") || "1D";
|
|
const limit = parseInt(url.searchParams.get("limit") || "30", 10);
|
|
console.log(`analyze/${ticker}: loader called with timeframe=${timeframe}, limit=${limit}`);
|
|
|
|
// Build base URL from request for server-side fetches
|
|
const reqUrl = new URL(request.url);
|
|
const host = request.headers.get("host") || reqUrl.host;
|
|
const protocol = reqUrl.protocol;
|
|
const baseUrl = `${protocol}//${host}`;
|
|
console.log(`analyze/${ticker}: baseUrl = ${baseUrl}`);
|
|
|
|
let position = null;
|
|
let orders = [];
|
|
let bars = [];
|
|
|
|
try {
|
|
// Fetch position
|
|
const posRes = await fetch(`${baseUrl}/api/alpaca/positions`);
|
|
console.log(`analyze/${ticker}: positions status = ${posRes.status}`);
|
|
const positions = posRes.ok ? await posRes.json() : [];
|
|
position = positions.find((p: any) => p.ticker === ticker)?.qty ?? null;
|
|
|
|
// Fetch orders
|
|
const ordRes = await fetch(`${baseUrl}/api/alpaca/orders`);
|
|
const ordersData = ordRes.ok ? await ordRes.json() : { orders: [] };
|
|
orders = ordersData.orders?.filter((o: any) => o.symbol === ticker) || [];
|
|
|
|
// Fetch bars for chart with timeframe and limit
|
|
console.log(`analyze/${ticker}: fetching bars from ${baseUrl}/api/alpaca/quote/${ticker}?timeframe=${timeframe}&limit=${limit}`);
|
|
const barsRes = await fetch(`${baseUrl}/api/alpaca/quote/${ticker}?timeframe=${timeframe}&limit=${limit}`);
|
|
console.log(`analyze/${ticker}: bars response status = ${barsRes.status}`);
|
|
const barsData = barsRes.ok ? await barsRes.json() : null;
|
|
console.log(`analyze/${ticker}: barsData =`, JSON.stringify(barsData));
|
|
bars = barsData?.bars || [];
|
|
} catch (err) {
|
|
console.error(`analyze/${ticker}: loader error`, err);
|
|
}
|
|
|
|
return Response.json({ ticker, position, orders, bars, timeframe, limit });
|
|
}
|
|
|
|
export default function StockDetail() {
|
|
const { ticker, position, orders, bars, timeframe, limit } = useLoaderData() as LoaderData;
|
|
const navigate = useNavigate();
|
|
const location = useLocation();
|
|
|
|
const updateParams = (newTimeframe: string, newLimit: number) => {
|
|
const searchParams = new URLSearchParams(location.search);
|
|
searchParams.set("timeframe", newTimeframe);
|
|
searchParams.set("limit", newLimit.toString());
|
|
navigate(`${location.pathname}?${searchParams.toString()}`, { replace: true });
|
|
};
|
|
|
|
// Convert Alpaca bars to TradingView format (YYYY-MM-DD for time)
|
|
const chartData = bars?.map((bar: any) => {
|
|
// Handle timestamp - could be string, number, or Date
|
|
let time = "";
|
|
if (bar.t) {
|
|
const date = new Date(bar.t);
|
|
if (!isNaN(date.getTime())) {
|
|
time = date.toISOString().split('T')[0];
|
|
}
|
|
}
|
|
return {
|
|
time,
|
|
open: bar.o,
|
|
high: bar.h,
|
|
low: bar.l,
|
|
close: bar.c,
|
|
};
|
|
}).filter((bar: any) => bar.time && bar.open != null && bar.high != null && bar.low != null && bar.close != null) || [];
|
|
|
|
console.log(`StockDetail: loaded ${bars?.length ?? 0} bars, transformed to ${chartData.length} chart points`);
|
|
|
|
return (
|
|
<div className="min-h-screen bg-gradient-to-br from-gray-50 to-blue-50">
|
|
<Navbar />
|
|
<div className="mx-auto max-w-7xl px-6 py-8">
|
|
<h1 className="text-3xl font-bold text-gray-900 mb-6">{ticker} Detail</h1>
|
|
|
|
<div className="bg-white rounded-xl shadow-lg p-6 mb-6 border border-gray-200">
|
|
<div className="flex items-center gap-4 mb-4">
|
|
<span className="text-gray-700 font-medium">Timeframe:</span>
|
|
<select
|
|
value={timeframe}
|
|
onChange={(e) => updateParams(e.target.value, limit)}
|
|
className="border border-gray-300 rounded-lg px-3 py-1.5 text-gray-900 focus:ring-2 focus:ring-blue-500"
|
|
>
|
|
{TIMEFRAMES.map((tf) => (
|
|
<option key={tf.value} value={tf.value}>{tf.label}</option>
|
|
))}
|
|
</select>
|
|
|
|
<span className="text-gray-700 font-medium">Bars:</span>
|
|
<select
|
|
value={limit}
|
|
onChange={(e) => updateParams(timeframe, parseInt(e.target.value))}
|
|
className="border border-gray-300 rounded-lg px-3 py-1.5 text-gray-900 focus:ring-2 focus:ring-blue-500"
|
|
>
|
|
<option value={10}>10</option>
|
|
<option value={30}>30</option>
|
|
<option value={50}>50</option>
|
|
<option value={100}>100</option>
|
|
</select>
|
|
</div>
|
|
|
|
<TradingViewChart ticker={ticker} data={chartData} />
|
|
</div>
|
|
|
|
<div className="mt-6 bg-white rounded-xl shadow-lg p-6 border border-gray-200">
|
|
<h2 className="text-xl font-bold text-gray-900 mb-4">Position</h2>
|
|
<p className="text-gray-600">{position ? `Quantity: ${position} shares` : "No position held"}</p>
|
|
</div>
|
|
|
|
<div className="mt-6 bg-white rounded-xl shadow-lg p-6 border border-gray-200">
|
|
<h2 className="text-xl font-bold text-gray-900 mb-4">Recent Orders</h2>
|
|
{orders.length === 0 ? (
|
|
<p className="text-gray-500">No orders found for {ticker}</p>
|
|
) : (
|
|
<div className="overflow-x-auto">
|
|
<table className="w-full text-sm">
|
|
<thead>
|
|
<tr className="border-b border-gray-200">
|
|
<th className="text-left py-2 px-3 font-medium text-gray-700">Side</th>
|
|
<th className="text-left py-2 px-3 font-medium text-gray-700">Qty</th>
|
|
<th className="text-left py-2 px-3 font-medium text-gray-700">Status</th>
|
|
<th className="text-left py-2 px-3 font-medium text-gray-700">Filled Price</th>
|
|
<th className="text-left py-2 px-3 font-medium text-gray-700">Filled At</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{orders.map((order: any, i: number) => (
|
|
<tr key={order.id || i} className="border-b border-gray-100">
|
|
<td className="py-2 px-3">
|
|
<span className={order.side === "buy" ? "text-green-600" : "text-red-600"}>
|
|
{order.side?.toUpperCase()}
|
|
</span>
|
|
</td>
|
|
<td className="py-2 px-3 text-gray-900">{order.qty}</td>
|
|
<td className="py-2 px-3 text-gray-900">{order.status}</td>
|
|
<td className="py-2 px-3 text-gray-900">
|
|
{order.filled_avg_price ? `$${parseFloat(order.filled_avg_price).toFixed(2)}` : "-"}
|
|
</td>
|
|
<td className="py-2 px-3 text-gray-600">
|
|
{order.filled_at ? new Date(order.filled_at).toLocaleDateString() : "-"}
|
|
</td>
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
} |