Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
55 changes: 55 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
# Dependencies
node_modules
.pnp
.pnp.js

# Testing
/coverage

# Next.js
.next/
out/
build
dist

# Environment Variables
.env
.env.local
.env.*
!.env.example
!.env.local.example

# Database
*.db
*.db-journal
*.sqlite
*.sqlite3
chat.db*

# Debug logs
npm-debug.log*
yarn-debug.log*
yarn-error.log*
.pnpm-debug.log*

# IDE/Editor
.idea/
.vscode/
*.swp
*.swo
.DS_Store

# Vercel
.vercel

# TypeScript
*.tsbuildinfo
next-env.d.ts

# Production builds
/build
/dist

# Cache
.cache/
node_modules
10 changes: 10 additions & 0 deletions app/agentstats/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
import AgentStats from '@/components/AgentStats';

export default function AgentsPage() {
return (
<div className="container mx-auto p-4">
<h1 className="text-2xl font-bold mb-6">Agent Statistics</h1>
<AgentStats />
</div>
);
}
84 changes: 42 additions & 42 deletions app/api/rooms/route.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,49 +2,49 @@ import { NextResponse } from "next/server";
import { listRooms, createRoom } from "@/server/store";
import { ChatRoom, ModelInfo } from "@/server/types";

// List all rooms
export async function GET(request: Request) {
try {
const { searchParams } = new URL(request.url);
const tags = searchParams.get("tags")?.split(",") || [];

const rooms = await listRooms(tags);
console.log('Available rooms:', rooms); // Debug log

return NextResponse.json({ rooms });
} catch (error) {
console.error('Error in GET /api/rooms:', error);
return NextResponse.json(
{ error: "Failed to fetch rooms", details: error },
{ status: 500 }
);
}
try {
const { searchParams } = new URL(request.url);
const tags = searchParams.get("tags")?.split(",") || [];

const rooms = await listRooms(tags);
console.log('Available rooms:', rooms);

return NextResponse.json({ rooms });
} catch (error) {
console.error('Error in GET /api/rooms:', error);
return NextResponse.json({
error: "Failed to fetch rooms",
message: error instanceof Error ? error.message : String(error),
stack: process.env.NODE_ENV === 'development' ? error instanceof Error ? error.stack : undefined : undefined
}, { status: 500 });
}
}

// Create a new room
export async function POST(request: Request) {
try {
const { name, topic, tags, creator } = await request.json() as {
name: string;
topic: string;
tags: string[];
creator: ModelInfo;
};

const room = await createRoom({
name,
topic,
tags,
participants: [creator],
createdAt: new Date().toISOString(),
messageCount: 0
});

return NextResponse.json({ room });
} catch (error) {
return NextResponse.json(
{ error: "Failed to create room" },
{ status: 500 }
);
}
}
try {
const { name, topic, tags, creator } = await request.json() as {
name: string;
topic: string;
tags: string[];
creator: ModelInfo;
};

const room = await createRoom({
name,
topic,
tags,
participants: [creator],
createdAt: new Date().toISOString(),
messageCount: 0
});

return NextResponse.json({ room });
} catch (error) {
return NextResponse.json({
error: "Failed to create room",
message: error instanceof Error ? error.message : String(error),
stack: process.env.NODE_ENV === 'development' ? error instanceof Error ? error.stack : undefined : undefined
}, { status: 500 });
}
}
185 changes: 185 additions & 0 deletions components/AgentStats.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,185 @@
'use client';

import { useEffect, useState } from 'react';
import { fetchAgents } from '@/server/actions';
import { ArrowUpIcon, ArrowDownIcon } from '@heroicons/react/20/solid';

interface Tweet {
tweetUrl: string;
tweetAuthorDisplayName: string;
smartEngagementPoints: number;
impressionsCount: number;
}

interface Contract {
chain: number;
contractAddress: string;
}

interface Agent {
agentName: string;
contracts: Contract[];
twitterUsernames: string[];
mindshare: number;
marketCap: number;
price: number;
volume24Hours: number;
holdersCount: number;
followersCount: number;
smartFollowersCount: number;
topTweets: Tweet[];
mindshareDeltaPercent: number;
marketCapDeltaPercent: number;
priceDeltaPercent: number;
volume24HoursDeltaPercent: number;
holdersCountDeltaPercent: number;
averageImpressionsCount: number;
averageImpressionsCountDeltaPercent: number;
averageEngagementsCount: number;
averageEngagementsCountDeltaPercent: number;
}

interface ApiResponse {
ok: {
data: Agent[];
currentPage: number;
totalPages: number;
totalCount: number;
};
}

function DeltaIndicator({ value }: { value: number }) {
if (value === 0) return null;
const isPositive = value > 0;
return (
<span className={`inline-flex items-center ml-2 ${isPositive ? 'text-green-500' : 'text-red-500'}`}>
{isPositive ? <ArrowUpIcon className="w-4 h-4" /> : <ArrowDownIcon className="w-4 h-4" />}
{Math.abs(value).toFixed(2)}%
</span>
);
}

export default function AgentStats() {
const [agents, setAgents] = useState<Agent[]>([]);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const [currentPage, setCurrentPage] = useState(1);
const [totalPages, setTotalPages] = useState(1);

useEffect(() => {
const loadAgents = async () => {
try {
const data = await fetchAgents(currentPage);
setAgents(data.ok.data);
setTotalPages(data.ok.totalPages);
} catch (err) {
setError(err instanceof Error ? err.message : 'Failed to fetch agents');
} finally {
setLoading(false);
}
};

loadAgents();
}, [currentPage]);

if (loading) return <div>Loading...</div>;
if (error) return <div>Error: {error}</div>;

return (
<div>
<div className="space-y-6">
{agents.map((agent) => (
<div key={agent.agentName} className="border rounded-lg p-4">
<h2 className="text-xl font-bold mb-2">{agent.agentName}</h2>

<div className="grid grid-cols-2 gap-4 mb-4">
<div className="space-y-2">
<div>
<span>Mindshare: {agent.mindshare.toFixed(2)}</span>
<DeltaIndicator value={agent.mindshareDeltaPercent} />
</div>

<div>
<span>Market Cap: ${(agent.marketCap / 1000000).toFixed(2)}M</span>
<DeltaIndicator value={agent.marketCapDeltaPercent} />
</div>

<div>
<span>Price: ${agent.price.toFixed(4)}</span>
<DeltaIndicator value={agent.priceDeltaPercent} />
</div>

<div>
<span>24h Volume: ${(agent.volume24Hours / 1000000).toFixed(2)}M</span>
<DeltaIndicator value={agent.volume24HoursDeltaPercent} />
</div>
</div>

<div className="space-y-2">
<div>
<span>Holders: {agent.holdersCount.toLocaleString()}</span>
<DeltaIndicator value={agent.holdersCountDeltaPercent} />
</div>

<div>
<span>Followers: {agent.followersCount.toLocaleString()}</span>
</div>

<div>
<span>Smart Followers: {agent.smartFollowersCount.toLocaleString()}</span>
</div>

<div>
<span>Avg Impressions: {agent.averageImpressionsCount.toLocaleString()}</span>
<DeltaIndicator value={agent.averageImpressionsCountDeltaPercent} />
</div>

<div>
<span>Avg Engagements: {agent.averageEngagementsCount.toFixed(1)}</span>
<DeltaIndicator value={agent.averageEngagementsCountDeltaPercent} />
</div>
</div>
</div>

<div>
<h3 className="font-semibold mb-2">Top Tweets</h3>
<div className="space-y-2">
{agent.topTweets.slice(0, 3).map((tweet, index) => (
<div key={index} className="text-sm">
<a href={tweet.tweetUrl} target="_blank" rel="noopener noreferrer"
className="text-blue-500 hover:underline">
{tweet.tweetAuthorDisplayName}
</a>
<span className="ml-2">
({tweet.smartEngagementPoints} points, {tweet.impressionsCount.toLocaleString()} impressions)
</span>
</div>
))}
</div>
</div>
</div>
))}
</div>

<div className="flex justify-between items-center mt-6">
<button
onClick={() => setCurrentPage(p => p - 1)}
disabled={currentPage === 1}
className="px-4 py-2 bg-primary text-primary-foreground rounded disabled:opacity-50"
>
Previous
</button>

<span>Page {currentPage} of {totalPages}</span>

<button
onClick={() => setCurrentPage(p => p + 1)}
disabled={currentPage === totalPages}
className="px-4 py-2 bg-primary text-primary-foreground rounded disabled:opacity-50"
>
Next
</button>
</div>
</div>
);
}
Loading