diff --git a/.env.example b/.env.example index c40911743d..0d808f655b 100644 --- a/.env.example +++ b/.env.example @@ -5,6 +5,11 @@ DISCORD_VOICE_CHANNEL_ID= # The ID of the voice channel the bot should join (opt # AI Model API Keys OPENAI_API_KEY= # OpenAI API key, starting with sk- +SMALL_OPENAI_MODEL= # Default: gpt-4o-mini +MEDIUM_OPENAI_MODEL= # Default: gpt-4o +LARGE_OPENAI_MODEL= # Default: gpt-4o +EMBEDDING_OPENAI_MODEL= # Default: text-embedding-3-small +IMAGE_OPENAI_MODEL= # Default: dall-e-3 # Eternal AI's Decentralized Inference API ETERNALAI_URL= @@ -48,6 +53,8 @@ TWITTER_EMAIL= # Account email TWITTER_2FA_SECRET= TWITTER_COOKIES= # Account cookies TWITTER_POLL_INTERVAL=120 # How often (in seconds) the bot should check for interactions +TWITTER_SEARCH_ENABLE=FALSE # Enable timeline search, WARNING this greatly increases your chance of getting banned +TWITTER_TARGET_USERS= # Comma separated list of Twitter user names to interact with X_SERVER_URL= XAI_API_KEY= XAI_MODEL= @@ -57,6 +64,10 @@ POST_INTERVAL_MIN= # Default: 90 POST_INTERVAL_MAX= # Default: 180 POST_IMMEDIATELY= +# Twitter action processing configuration +ACTION_INTERVAL=300000 # Interval in milliseconds between action processing runs (default: 5 minutes) +ENABLE_ACTION_PROCESSING=false # Set to true to enable the action processing loop + # Feature Flags IMAGE_GEN= # Set to TRUE to enable image generation USE_OPENAI_EMBEDDING= # Set to TRUE for OpenAI/1536, leave blank for local @@ -107,6 +118,10 @@ LLAMALOCAL_PATH= # Default: "" which is the current directory in plugin-node/dis # API Keys ANTHROPIC_API_KEY= # For Claude +SMALL_ANTHROPIC_MODEL= # Default: claude-3-haiku-20240307 +MEDIUM_ANTHROPIC_MODEL= # Default: claude-3-5-sonnet-20241022 +LARGE_ANTHROPIC_MODEL= # Default: claude-3-5-sonnet-20241022 + HEURIST_API_KEY= # Get from https://heurist.ai/dev-access # Heurist Models @@ -210,6 +225,12 @@ WALLET_SECRET_SALT= # ONLY DEFINE IF YOU WANT TO USE TEE Plugin, otherwise it wi # Galadriel Configuration GALADRIEL_API_KEY=gal-* # Get from https://dashboard.galadriel.com/ +# Venice Configuration +VENICE_API_KEY= # generate from venice settings +SMALL_VENICE_MODEL= # Default: llama-3.3-70b +MEDIUM_VENICE_MODEL= # Default: llama-3.3-70b +LARGE_VENICE_MODEL= # Default: llama-3.1-405b + # fal.ai Configuration FAL_API_KEY= FAL_AI_LORA_PATH= @@ -235,6 +256,13 @@ INTERNET_COMPUTER_ADDRESS= APTOS_PRIVATE_KEY= # Aptos private key APTOS_NETWORK= # must be one of mainnet, testnet +# EchoChambers Configuration +ECHOCHAMBERS_API_URL=http://127.0.0.1:3333 +ECHOCHAMBERS_API_KEY=testingkey0011 +ECHOCHAMBERS_USERNAME=eliza +ECHOCHAMBERS_DEFAULT_ROOM=general +ECHOCHAMBERS_POLL_INTERVAL=60 +ECHOCHAMBERS_MAX_MESSAGES=10 # AWS S3 Configuration Settings for File Upload AWS_ACCESS_KEY_ID= @@ -242,3 +270,7 @@ AWS_SECRET_ACCESS_KEY= AWS_REGION= AWS_S3_BUCKET= AWS_S3_UPLOAD_PATH= + + +# Deepgram +DEEPGRAM_API_KEY= diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index 535617fb44..58c6cdcc55 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -34,7 +34,7 @@ jobs: echo "NODE_ENV=test" >> packages/core/.env.test - name: Run tests - run: cd packages/core && pnpm test + run: cd packages/core && pnpm test:coverage - name: Build packages run: pnpm run build diff --git a/.github/workflows/integrationTests.yaml b/.github/workflows/integrationTests.yaml new file mode 100644 index 0000000000..86f1b6f20e --- /dev/null +++ b/.github/workflows/integrationTests.yaml @@ -0,0 +1,55 @@ +name: integration-test +on: + push: + branches: + - "*" + pull_request_target: + branches: + - "*" +jobs: + smoke-tests: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + + - uses: pnpm/action-setup@v3 + with: + version: 9.4.0 + + - uses: actions/setup-node@v4 + with: + node-version: "23" + cache: "pnpm" + + - name: Run smoke tests + run: pnpm run smokeTests + integration-tests: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + + - uses: pnpm/action-setup@v3 + with: + version: 9.4.0 + + - uses: actions/setup-node@v4 + with: + node-version: "23" + cache: "pnpm" + + - name: Install dependencies + run: pnpm install -r + + - name: Build packages + run: pnpm build + + - name: Run integration tests + env: + OPENAI_API_KEY: ${{ secrets.OPENAI_API_KEY }} + run: | + if [ -z "$OPENAI_API_KEY" ]; then + echo "Skipping integration tests due to missing required API keys" + exit 1 + else + pnpm run integrationTests + fi diff --git a/.github/workflows/require-develop.yml b/.github/workflows/require-develop.yml new file mode 100644 index 0000000000..4519a3b5ca --- /dev/null +++ b/.github/workflows/require-develop.yml @@ -0,0 +1,18 @@ +name: Check pull request source branch +on: + pull_request_target: + types: + - opened + - reopened + - synchronize + - edited +jobs: + check-branches: + runs-on: ubuntu-latest + steps: + - name: Check branches + run: | + if [ ${{ github.head_ref }} != "develop" ] && [ ${{ github.base_ref }} == "main" ]; then + echo "Merge requests to main branch are only allowed from dev branch." + exit 1 + fi diff --git a/.gitignore b/.gitignore index 83c6975909..9035e26223 100644 --- a/.gitignore +++ b/.gitignore @@ -46,3 +46,5 @@ packages/plugin-coinbase/package-lock.json tsup.config.bundled_*.mjs .turbo + +coverage \ No newline at end of file diff --git a/CHANGELOG.md b/CHANGELOG.md index 4801b48eb5..f32c8efe0d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,121 @@ # Changelog +## [v0.1.6-alpha.1](https://github.com/ai16z/eliza/tree/v0.1.6-alpha.1) (2024-12-13) + +[Full Changelog](https://github.com/ai16z/eliza/compare/v0.1.5-alpha.5...v0.1.6-alpha.1) + +**Implemented enhancements:** + +- Add Venice.ai Model Provider [\#1016](https://github.com/ai16z/eliza/issues/1016) +- Need to add media file upload for posting tweets with image from imageGenerationPlugin. Currently only discord has this implemented [\#969](https://github.com/ai16z/eliza/issues/969) +- Script to create core memories for the agent [\#967](https://github.com/ai16z/eliza/issues/967) +- feat: add hot-reloading for agent dependencies [\#930](https://github.com/ai16z/eliza/issues/930) +- Improve `dev.sh` Script to Enhance Plugin Development Workflow [\#888](https://github.com/ai16z/eliza/issues/888) + +**Fixed bugs:** + +- How to set the model class for Anthropic? [\#988](https://github.com/ai16z/eliza/issues/988) +- Twitter Search Client Broken [\#943](https://github.com/ai16z/eliza/issues/943) +- Stuck querying when @'ing it in Discord [\#921](https://github.com/ai16z/eliza/issues/921) +- Error pnpm start - Promise.withResolvers\(\): pdfjs-dist [\#902](https://github.com/ai16z/eliza/issues/902) +- Running tests on start and dev? [\#893](https://github.com/ai16z/eliza/issues/893) +- build: eliza docs build creates 130 files that want to be modified/added to git [\#849](https://github.com/ai16z/eliza/issues/849) +- EVM Plugin can't run any action [\#735](https://github.com/ai16z/eliza/issues/735) +- Bug: plugin-solana crash report [\#467](https://github.com/ai16z/eliza/issues/467) + +**Closed issues:** + +- Issue: Unable to Post Tweets Using Eliza Integration with Twitter via Cookies or OAuth2 [\#901](https://github.com/ai16z/eliza/issues/901) + +**Merged pull requests:** + +- chore: release develop into main [\#1045](https://github.com/ai16z/eliza/pull/1045) ([odilitime](https://github.com/odilitime)) +- fix: re-enable generateNewTweetLoop / lint fixes [\#1043](https://github.com/ai16z/eliza/pull/1043) ([odilitime](https://github.com/odilitime)) +- docs: characterfile.md docs outdated with latest eliza version [\#1042](https://github.com/ai16z/eliza/pull/1042) ([tqdpham96](https://github.com/tqdpham96)) +- feat: Add AI Agent Dev School Tutorial Link [\#1038](https://github.com/ai16z/eliza/pull/1038) ([lalalune](https://github.com/lalalune)) +- fix: use pull\_request\_target for integration tests [\#1035](https://github.com/ai16z/eliza/pull/1035) ([jnaulty](https://github.com/jnaulty)) +- feat: Add Discord Team features [\#1032](https://github.com/ai16z/eliza/pull/1032) ([azep-ninja](https://github.com/azep-ninja)) +- feat: client-discord stop implementation / agent improvements [\#1029](https://github.com/ai16z/eliza/pull/1029) ([odilitime](https://github.com/odilitime)) +- chore: Push Develop into Main [\#1028](https://github.com/ai16z/eliza/pull/1028) ([odilitime](https://github.com/odilitime)) +- feat: improve voice processing and add deepgram transcription option [\#1026](https://github.com/ai16z/eliza/pull/1026) ([tcm390](https://github.com/tcm390)) +- docs: Update README.md [\#1025](https://github.com/ai16z/eliza/pull/1025) ([sergical](https://github.com/sergical)) +- docs: Update README.md [\#1024](https://github.com/ai16z/eliza/pull/1024) ([sergical](https://github.com/sergical)) +- chore: Twitter fetchHomeTimeline rework [\#1021](https://github.com/ai16z/eliza/pull/1021) ([odilitime](https://github.com/odilitime)) +- chore: Update CI configuration to enable test coverage and add covera… [\#1019](https://github.com/ai16z/eliza/pull/1019) ([snobbee](https://github.com/snobbee)) +- docs: "AI Agent Dev School Part 4" livestream notes [\#1015](https://github.com/ai16z/eliza/pull/1015) ([YoungPhlo](https://github.com/YoungPhlo)) +- docs: Add templates documentation to the project [\#1013](https://github.com/ai16z/eliza/pull/1013) ([Lukapetro](https://github.com/Lukapetro)) +- feat: Add custom fetch logic for agent [\#1010](https://github.com/ai16z/eliza/pull/1010) ([v1xingyue](https://github.com/v1xingyue)) +- feat: Plugin evm multichain [\#1009](https://github.com/ai16z/eliza/pull/1009) ([nicky-ru](https://github.com/nicky-ru)) +- feat: add venice.ai api model provider [\#1008](https://github.com/ai16z/eliza/pull/1008) ([proteanx](https://github.com/proteanx)) +- feat: improve Twitter client with action processing [\#1007](https://github.com/ai16z/eliza/pull/1007) ([dorianjanezic](https://github.com/dorianjanezic)) +- chore: Bring Develop up to date with HEAD [\#1006](https://github.com/ai16z/eliza/pull/1006) ([odilitime](https://github.com/odilitime)) +- feat: create example folder with example plugin [\#1004](https://github.com/ai16z/eliza/pull/1004) ([monilpat](https://github.com/monilpat)) +- chore: Twitter search switch [\#1003](https://github.com/ai16z/eliza/pull/1003) ([odilitime](https://github.com/odilitime)) +- fix: add callback to action in farcaster client [\#1002](https://github.com/ai16z/eliza/pull/1002) ([sin-bufan](https://github.com/sin-bufan)) +- fix: typo initialize [\#1000](https://github.com/ai16z/eliza/pull/1000) ([cryptofish7](https://github.com/cryptofish7)) +- feat: allow users to configure models for openai and anthropic [\#999](https://github.com/ai16z/eliza/pull/999) ([oxSaturn](https://github.com/oxSaturn)) +- add echochambers [\#997](https://github.com/ai16z/eliza/pull/997) ([savageops](https://github.com/savageops)) +- test: adding parsing tests. changed files parsing.test.ts [\#996](https://github.com/ai16z/eliza/pull/996) ([ai16z-demirix](https://github.com/ai16z-demirix)) +- feat: create README\_DE.md [\#995](https://github.com/ai16z/eliza/pull/995) ([GottliebFreudenreich](https://github.com/GottliebFreudenreich)) +- fix: Fix Twitter Search Logic and Add Galadriel Image Model [\#994](https://github.com/ai16z/eliza/pull/994) ([dontAskVI](https://github.com/dontAskVI)) +- test: Initial release of smoke/integration tests + testing framework [\#993](https://github.com/ai16z/eliza/pull/993) ([jzvikart](https://github.com/jzvikart)) +- fix: a typo in characterfile.md [\#986](https://github.com/ai16z/eliza/pull/986) ([oxSaturn](https://github.com/oxSaturn)) +- fix: Goat Plugin + AWS S3 Service error when env vars absent [\#985](https://github.com/ai16z/eliza/pull/985) ([jnaulty](https://github.com/jnaulty)) +- docs: add WSL Setup Guide to documentation [\#983](https://github.com/ai16z/eliza/pull/983) ([ileana-pr](https://github.com/ileana-pr)) +- fix: docker trying to filter out missing docs package [\#978](https://github.com/ai16z/eliza/pull/978) ([odilitime](https://github.com/odilitime)) +- chore: fix broken lockfile [\#977](https://github.com/ai16z/eliza/pull/977) ([shakkernerd](https://github.com/shakkernerd)) +- chore: add how to startup chat ui [\#976](https://github.com/ai16z/eliza/pull/976) ([yodamaster726](https://github.com/yodamaster726)) +- feat: Add hyperbolic env vars to override model class [\#974](https://github.com/ai16z/eliza/pull/974) ([meppsilon](https://github.com/meppsilon)) +- LinkedIn Client [\#973](https://github.com/ai16z/eliza/pull/973) ([bkellgren](https://github.com/bkellgren)) +- Fix farcaster client process action issue [\#963](https://github.com/ai16z/eliza/pull/963) ([sin-bufan](https://github.com/sin-bufan)) +- fix\(agent\): correct EVM plugin activation condition [\#962](https://github.com/ai16z/eliza/pull/962) ([0xAsten](https://github.com/0xAsten)) +- fix: use MAX\_TWEET\_LENGTH from setting [\#960](https://github.com/ai16z/eliza/pull/960) ([oxSaturn](https://github.com/oxSaturn)) +- fix: Revert "docs: add WSL installation guide" [\#959](https://github.com/ai16z/eliza/pull/959) ([monilpat](https://github.com/monilpat)) +- feat: add dev script to plugin-aptos [\#956](https://github.com/ai16z/eliza/pull/956) ([asianviking](https://github.com/asianviking)) +- chore: rename intiface plugin [\#955](https://github.com/ai16z/eliza/pull/955) ([odilitime](https://github.com/odilitime)) +- fix: revert llamacloud endpoint change [\#954](https://github.com/ai16z/eliza/pull/954) ([odilitime](https://github.com/odilitime)) +- feat: allow character.json settings models for open router [\#953](https://github.com/ai16z/eliza/pull/953) ([odilitime](https://github.com/odilitime)) +- chore: 947 add other evm chains to wallet [\#949](https://github.com/ai16z/eliza/pull/949) ([n00b21337](https://github.com/n00b21337)) +- fix: telegram response memory userId to agentId [\#948](https://github.com/ai16z/eliza/pull/948) ([bmgalego](https://github.com/bmgalego)) +- docs: add WSL installation guide [\#946](https://github.com/ai16z/eliza/pull/946) ([ileana-pr](https://github.com/ileana-pr)) +- feat: Supports upload files to AWS S3. [\#941](https://github.com/ai16z/eliza/pull/941) ([xwxtwd](https://github.com/xwxtwd)) +- feat: process all responses actions [\#940](https://github.com/ai16z/eliza/pull/940) ([bmgalego](https://github.com/bmgalego)) +- feat: add callback handler to runtime evaluate method [\#938](https://github.com/ai16z/eliza/pull/938) ([bmgalego](https://github.com/bmgalego)) +- fix: update package name in faq [\#937](https://github.com/ai16z/eliza/pull/937) ([oxSaturn](https://github.com/oxSaturn)) +- fix: update quickstart and .env.example [\#932](https://github.com/ai16z/eliza/pull/932) ([oxSaturn](https://github.com/oxSaturn)) +- feat: add dynamic watch paths for agent development [\#931](https://github.com/ai16z/eliza/pull/931) ([samuveth](https://github.com/samuveth)) +- feat: flow update generate object [\#929](https://github.com/ai16z/eliza/pull/929) ([btspoony](https://github.com/btspoony)) +- feat: Config eternalai model from env [\#927](https://github.com/ai16z/eliza/pull/927) ([genesis-0000](https://github.com/genesis-0000)) +- feat: Add NanoGPT provider [\#926](https://github.com/ai16z/eliza/pull/926) ([dylan1951](https://github.com/dylan1951)) +- fix: use of Heurist model env vars [\#924](https://github.com/ai16z/eliza/pull/924) ([boxhock](https://github.com/boxhock)) +- feat: add readContract / invokeContract functionality to Coinbase plugin [\#923](https://github.com/ai16z/eliza/pull/923) ([monilpat](https://github.com/monilpat)) +- chore: deprecate text based way of generating JSON [\#920](https://github.com/ai16z/eliza/pull/920) ([monilpat](https://github.com/monilpat)) +- feat: create README\_TH.md [\#918](https://github.com/ai16z/eliza/pull/918) ([asianviking](https://github.com/asianviking)) +- feat: update gaianet config [\#915](https://github.com/ai16z/eliza/pull/915) ([L-jasmine](https://github.com/L-jasmine)) +- fix: Farcater client cleanup and fixed response logic [\#914](https://github.com/ai16z/eliza/pull/914) ([sayangel](https://github.com/sayangel)) +- Twitter client enhancements [\#913](https://github.com/ai16z/eliza/pull/913) ([tharak123455](https://github.com/tharak123455)) +- feat: MAX\_TWEET\_LENGTH env implementation [\#912](https://github.com/ai16z/eliza/pull/912) ([onur-saf](https://github.com/onur-saf)) +- feat: allow users to configure models for groq [\#910](https://github.com/ai16z/eliza/pull/910) ([oxSaturn](https://github.com/oxSaturn)) +- fix: evaluation json parsing [\#907](https://github.com/ai16z/eliza/pull/907) ([cygaar](https://github.com/cygaar)) +- fix: twitter actions not triggering [\#903](https://github.com/ai16z/eliza/pull/903) ([cygaar](https://github.com/cygaar)) +- chore: Consistent language for Community & Contact link label [\#899](https://github.com/ai16z/eliza/pull/899) ([golryang](https://github.com/golryang)) +- chore: pass env variables when setting up GOAT and update GOAT readme [\#898](https://github.com/ai16z/eliza/pull/898) ([0xaguspunk](https://github.com/0xaguspunk)) +- docs: Add What Did You Get Done This Week \#4 summaries and timestamps [\#895](https://github.com/ai16z/eliza/pull/895) ([YoungPhlo](https://github.com/YoungPhlo)) +- chore: improved dev command [\#892](https://github.com/ai16z/eliza/pull/892) ([shakkernerd](https://github.com/shakkernerd)) +- chore: added more help message to the important notice text. [\#891](https://github.com/ai16z/eliza/pull/891) ([shakkernerd](https://github.com/shakkernerd)) +- chore: update models for groq [\#890](https://github.com/ai16z/eliza/pull/890) ([oxSaturn](https://github.com/oxSaturn)) +- Feat : github image cicd [\#889](https://github.com/ai16z/eliza/pull/889) ([v1xingyue](https://github.com/v1xingyue)) +- chore: enhance dev script, performance improvement and add help message [\#887](https://github.com/ai16z/eliza/pull/887) ([shakkernerd](https://github.com/shakkernerd)) +- chore: disable building docs on build command [\#884](https://github.com/ai16z/eliza/pull/884) ([shakkernerd](https://github.com/shakkernerd)) +- fix: re-enable coverage report upload to Codecov in CI workflow [\#880](https://github.com/ai16z/eliza/pull/880) ([snobbee](https://github.com/snobbee)) +- feat: Add Flow Blockchain plugin [\#874](https://github.com/ai16z/eliza/pull/874) ([btspoony](https://github.com/btspoony)) +- feat: Add TEE Mode to Solana Plugin [\#835](https://github.com/ai16z/eliza/pull/835) ([HashWarlock](https://github.com/HashWarlock)) +- feat: add hyperbolic api to eliza [\#828](https://github.com/ai16z/eliza/pull/828) ([meppsilon](https://github.com/meppsilon)) +- loading indicator [\#827](https://github.com/ai16z/eliza/pull/827) ([tcm390](https://github.com/tcm390)) +- use github access token [\#825](https://github.com/ai16z/eliza/pull/825) ([tcm390](https://github.com/tcm390)) +- fix: refactor contributor page [\#809](https://github.com/ai16z/eliza/pull/809) ([tcm390](https://github.com/tcm390)) +- feat: implement advanced coinbase trading [\#725](https://github.com/ai16z/eliza/pull/725) ([monilpat](https://github.com/monilpat)) + ## [v0.1.5-alpha.5](https://github.com/ai16z/eliza/tree/v0.1.5-alpha.5) (2024-12-07) [Full Changelog](https://github.com/ai16z/eliza/compare/v0.1.5-alpha.4...v0.1.5-alpha.5) diff --git a/README.md b/README.md index b87419cfad..e187cc8c4a 100644 --- a/README.md +++ b/README.md @@ -25,6 +25,9 @@ - ☁️ Supports many models (local Llama, OpenAI, Anthropic, Groq, etc.) - 📦 Just works! +## Video Tutorials +[AI Agent Dev School](https://www.youtube.com/watch?v=ArptLpQiKfI&list=PLx5pnFXdPTRzWla0RaOxALTSTnVq53fKL) + ## 🎯 Use Cases - 🤖 Chatbots @@ -50,7 +53,7 @@ git clone https://github.com/ai16z/eliza-starter.git cp .env.example .env -pnpm i && pnpm start +pnpm i && pnpm build && pnpm start ``` Then read the [Documentation](https://ai16z.github.io/eliza/) to learn how to customize your Eliza. @@ -90,7 +93,7 @@ sh scripts/start.sh ### Edit the character file -1. Open `agent/src/character.ts` to modify the default character. Uncomment and edit. +1. Open `packages/core/src/defaultCharacter.ts` to modify the default character. Uncomment and edit. 2. To load custom characters: - Use `pnpm start --characters="path/to/your/character.json"` diff --git a/agent/src/index.ts b/agent/src/index.ts index 9e030d3557..cd92b6005d 100644 --- a/agent/src/index.ts +++ b/agent/src/index.ts @@ -60,6 +60,12 @@ export const wait = (minTime: number = 1000, maxTime: number = 3000) => { return new Promise((resolve) => setTimeout(resolve, waitTime)); }; +const logFetch = async (url: string, options: any) => { + elizaLogger.info(`Fetching ${url}`); + elizaLogger.info(options); + return fetch(url, options); +}; + export function parseArguments(): { character?: string; characters?: string; @@ -280,6 +286,11 @@ export function getTokenForProvider( character.settings?.secrets?.HYPERBOLIC_API_KEY || settings.HYPERBOLIC_API_KEY ); + case ModelProviderName.VENICE: + return ( + character.settings?.secrets?.VENICE_API_KEY || + settings.VENICE_API_KEY + ); } } @@ -312,41 +323,53 @@ function initializeDatabase(dataDir: string) { } } +// also adds plugins from character file into the runtime export async function initializeClients( character: Character, runtime: IAgentRuntime ) { - const clients = []; - const clientTypes = + // each client can only register once + // and if we want two we can explicitly support it + const clients: Record = {}; + const clientTypes:string[] = character.clients?.map((str) => str.toLowerCase()) || []; + elizaLogger.log('initializeClients', clientTypes, 'for', character.name) if (clientTypes.includes("auto")) { const autoClient = await AutoClientInterface.start(runtime); - if (autoClient) clients.push(autoClient); + if (autoClient) clients.auto = autoClient; } if (clientTypes.includes("discord")) { - clients.push(await DiscordClientInterface.start(runtime)); + const discordClient = await DiscordClientInterface.start(runtime); + if (discordClient) clients.discord = discordClient; } if (clientTypes.includes("telegram")) { const telegramClient = await TelegramClientInterface.start(runtime); - if (telegramClient) clients.push(telegramClient); + if (telegramClient) clients.telegram = telegramClient; } if (clientTypes.includes("twitter")) { - const twitterClients = await TwitterClientInterface.start(runtime); - clients.push(twitterClients); + TwitterClientInterface.enableSearch = !isFalsish(getSecret(character, "TWITTER_SEARCH_ENABLE")); + const twitterClient = await TwitterClientInterface.start(runtime); + if (twitterClient) clients.twitter = twitterClient; } if (clientTypes.includes("farcaster")) { - const farcasterClients = new FarcasterAgentClient(runtime); - farcasterClients.start(); - clients.push(farcasterClients); + // why is this one different :( + const farcasterClient = new FarcasterAgentClient(runtime); + if (farcasterClient) { + farcasterClient.start(); + clients.farcaster = farcasterClient; + } } + elizaLogger.log('client keys', Object.keys(clients)); + if (character.plugins?.length > 0) { for (const plugin of character.plugins) { + // if plugin has clients, add those.. if (plugin.clients) { for (const client of plugin.clients) { clients.push(await client.start(runtime)); @@ -358,8 +381,24 @@ export async function initializeClients( return clients; } +function isFalsish(input: any): boolean { + // If the input is exactly NaN, return true + if (Number.isNaN(input)) { + return true; + } + + // Convert input to a string if it's not null or undefined + const value = input == null ? '' : String(input); + + // List of common falsish string representations + const falsishValues = ['false', '0', 'no', 'n', 'off', 'null', 'undefined', '']; + + // Check if the value (trimmed and lowercased) is in the falsish list + return falsishValues.includes(value.trim().toLowerCase()); +} + function getSecret(character: Character, secret: string) { - return character.settings.secrets?.[secret] || process.env[secret]; + return character.settings?.secrets?.[secret] || process.env[secret]; } let nodePlugin: any | undefined; @@ -369,7 +408,7 @@ export async function createAgent( db: IDatabaseAdapter, cache: ICacheManager, token: string -) { +):AgentRuntime { elizaLogger.success( elizaLogger.successesTitle, "Creating runtime for character", @@ -402,6 +441,7 @@ export async function createAgent( modelProvider: character.modelProvider, evaluators: [], character, + // character.plugins are handled when clients are added plugins: [ bootstrapPlugin, getSecret(character, "CONFLUX_CORE_PRIVATE_KEY") @@ -456,22 +496,23 @@ export async function createAgent( services: [], managers: [], cacheManager: cache, + fetch: logFetch, }); } -function intializeFsCache(baseDir: string, character: Character) { +function initializeFsCache(baseDir: string, character: Character) { const cacheDir = path.resolve(baseDir, character.id, "cache"); const cache = new CacheManager(new FsCacheAdapter(cacheDir)); return cache; } -function intializeDbCache(character: Character, db: IDatabaseCacheAdapter) { +function initializeDbCache(character: Character, db: IDatabaseCacheAdapter) { const cache = new CacheManager(new DbCacheAdapter(db, character.id)); return cache; } -async function startAgent(character: Character, directClient) { +async function startAgent(character: Character, directClient):AgentRuntime { let db: IDatabaseAdapter & IDatabaseCacheAdapter; try { character.id ??= stringToUuid(character.name); @@ -489,16 +530,22 @@ async function startAgent(character: Character, directClient) { await db.init(); - const cache = intializeDbCache(character, db); - const runtime = await createAgent(character, db, cache, token); + const cache = initializeDbCache(character, db); + const runtime:AgentRuntime = await createAgent(character, db, cache, token); + // start services/plugins/process knowledge await runtime.initialize(); - const clients = await initializeClients(character, runtime); + // start assigned clients + runtime.clients = await initializeClients(character, runtime); + // add to container directClient.registerAgent(runtime); - return clients; + // report to console + elizaLogger.debug(`Started ${character.name} as ${runtime.agentId}`) + + return runtime; } catch (error) { elizaLogger.error( `Error starting agent for character ${character.name}:`, @@ -542,8 +589,8 @@ const startAgents = async () => { }); } - elizaLogger.log("Chat started. Type 'exit' to quit."); if (!args["non-interactive"]) { + elizaLogger.log("Chat started. Type 'exit' to quit."); chat(); } }; diff --git a/docs/community/Streams/12-2024/2024-12-10.md b/docs/community/Streams/12-2024/2024-12-10.md new file mode 100644 index 0000000000..51afc2133f --- /dev/null +++ b/docs/community/Streams/12-2024/2024-12-10.md @@ -0,0 +1,94 @@ +--- +sidebar_position: 4 +title: "AI Agent Dev School Part 4" +description: "AI Pizza: Hacking Eliza for Domino's Delivery (plus TEE Deep Dive)" +--- + +# AI Agent Dev School Part 4 + +**AI Pizza: Hacking Eliza for Domino's Delivery (plus TEE Deep Dive)** + +Date: 2024-12-10 +YouTube Link: https://www.youtube.com/watch?v=6I9e9pJprDI + +## Timestamps + +Part 1: Trusted Execution Environments (TEEs) with Agent Joshua +- **00:00:09** - Stream starts, initial setup issues. +- **00:01:58** - Intro to Trusted Execution Environments (TEEs). +- **00:08:03** - Agent Joshua begins explaining TEEs and the Eliza plugin. +- **00:19:15** - Deeper dive into remote attestation. +- **00:24:50** - Discussion of derived keys. +- **00:37:00** - Deploying to a real TEE, Phala Network's TEE cloud. +- **00:50:48** - Q&A with Joshua, contact info, and next steps. + +Part 2: Building a Domino's pizza ordering agent +- **01:04:37** - Transition to building a Domino's pizza ordering agent. +- **01:14:20** - Discussion of the pizza ordering agent’s order flow and state machine. +- **01:22:07** - Using Claude to generate a state machine diagram. +- **01:32:17** - Creating the Domino's plugin in Eliza. +- **01:54:15** - Working on the pizza order provider. +- **02:16:46** - Pizza provider code completed. +- **02:28:50** - Discussion of caching customer and order data. +- **03:13:45** - Pushing fixes to main branch and continuing work on the agent. +- **04:24:30** - Discussion of summarizing past agent dev school sessions. +- **05:01:18** - Shaw returns, admits to ordering Domino's manually. +- **05:09:00** - Discussing payment flow and a confirm order action. +- **05:27:17** - Final code push, wrap-up, and end of stream. + + +## Summary + +This is a livestream titled "AI Agent Dev School Part 4" from the ai16z project, featuring Shaw. The stream is divided into two main parts: a technical discussion on Trusted Execution Environments (TEEs) and a coding session where Shaw attempts to build a pizza-ordering agent using the Domino's API. + +**Part 1: Trusted Execution Environments (TEEs) with Agent Joshua** + +This segment begins with Shaw introducing the concept of TEEs and their importance for running autonomous agents securely. He emphasizes the need to protect private keys and ensure that code execution is tamper-proof. Joshua from the Phala Network is brought on to explain TEEs in more detail and demonstrate how to use the TEE plugin he built for Eliza. + +* **Key Concepts:** + * **Trusted Execution Environments (TEEs):** Secure areas within a processor that isolate code and data, protecting them from unauthorized access and tampering. + * **Secure Enclave:** A cryptographic primitive that allows data to be encrypted and isolated within a processor. + * **Remote Attestation:** A method to verify that a program running inside a TEE is genuine and hasn't been tampered with, providing verifiability to users. + * **D-Stack:** An SDK developed in collaboration with Flashbots and Andrew Miller, enabling developers to build and launch Docker containers in TEEs. + * **Derived Key Provider:** A component that generates cryptographic keys based on a secret salt, ensuring that private keys are not exposed to humans. + +* **Demonstration:** + * Joshua walks through the process of setting up and deploying an Eliza agent in a TEE simulator, demonstrating how to generate remote attestations and derive keys. + * He shows how to use the remote attestation explorer to verify the authenticity of the agent running inside the TEE. + * He explains how to build a Docker image of the agent and deploy it to the Phala Network's TEE cloud solution. + +* **Use Cases:** + * Securely storing private keys for on-chain actions. + * Ensuring the integrity of autonomous agents, preventing tampering or unauthorized access. + * Providing verifiable execution for users and investors. + +* **Phala Network's TEE Cloud:** + * Joshua introduces Phala Network's TEE cloud solution, which allows developers to deploy Docker images and host their agents in a trusted execution environment. + * He mentions that the service supports various compute-intensive applications beyond AI agents. + * He invites interested developers to contact him on Discord (@hashwarlock) for onboarding and further details. + +**Part 2: Building a Pizza Ordering Agent** + +In the second part, Shaw transitions to a more lighthearted coding session where he attempts to build an agent that can order a pizza using the Domino's API. He highlights the challenges of handling payments securely and connecting user information to the conversation. + +* **Challenges:** + * Securely handling payment information. + * Connecting user data to the current conversation. + * Defining the order flow using a state machine. + +* **Approach:** + * Shaw uses a state machine to model the pizza ordering process, defining different states and transitions based on user input and available information. + * He uses Claude (an AI assistant) to generate code snippets and assist with the development process. + * He decides to initially focus on a simplified version where the user's payment information is hardcoded in the environment variables, and the agent only needs to collect the user's address. + +## Hot Takes + +1. **"Maybe we'll mix it on LinkedIn so people can order Domino's on LinkedIn. There you go. Now we're cooking." (00:03:26)** - Shaw's seemingly flippant idea of ordering pizza on LinkedIn highlights the potential for integrating everyday services into unexpected platforms through agents. This sparked discussion about the wider implications for businesses and social media. + +2. **"Yeah, it'll probably get drained real quick. These fucking people." (00:28:30)** - Shaw accidentally leaked an API key on stream and expressed frustration with viewers who noticed, exposing the real-world risks of handling sensitive information during development, especially in a live environment. + +3. **"The secret to making a billion dollars is to use the existing agent framework to deliver apps to people on social media that they want." (01:09:35)** - Shaw’s strong assertion about focusing on building apps *using* existing frameworks rather than creating new ones is a bold statement about the current agent development landscape, suggesting that innovation lies in application development, not framework creation. + +4. **"So those are like, honest to God, if the bots are better than like 70% of tweets on Twitter, they're better than like 99.7 tweets and posts on LinkedIn." (01:39:57)** - This provocative comparison of content quality between Twitter and LinkedIn, suggesting bots surpass most LinkedIn posts, fueled lively debate in the chat and raised questions about the role and value of human-generated content in the age of AI. + +5. **"I subliminally messaged Domino's into my own brain, and now I have to eat it." (05:01:24)** - After hours of working on the pizza bot, Shaw abandoned the live coding attempt and ordered pizza manually, a humorous but relatable moment that highlighted the challenges and frustrations of software development, even when aided by AI. It also underscores the human desire for immediate gratification, even in the face of a potentially groundbreaking technological advancement. diff --git a/docs/community/components/Accordion.tsx b/docs/community/components/Accordion.tsx new file mode 100644 index 0000000000..9b4748089e --- /dev/null +++ b/docs/community/components/Accordion.tsx @@ -0,0 +1,183 @@ +import React, { useState, useRef, useEffect } from "react"; +import { GitHubItem } from "./Contributions"; +import { GITHUB_PAGE_LIMIT } from "./Contributors"; + +interface AccordionProps { + title: string; + isOpen: boolean; + onToggle: () => void; + data: GitHubItem[]; + loadMore?: () => void; + total_count: number; + primaryText?: string; + secondaryText?: string; + mainBackgroundColor?: string; +} + +export const Accordion: React.FC = ({ + title, + isOpen, + onToggle, + data, + loadMore, + total_count, + primaryText, + secondaryText, + mainBackgroundColor, +}) => { + const [hoveredIndex, setHoveredIndex] = useState(null); + const [hoverLoadMore, setHoverLoadMore] = useState(false); + const [maxHeight, setMaxHeight] = useState( + isOpen ? "1000px" : "0px", + ); + + const contentRef = useRef(null); + + React.useEffect(() => { + setMaxHeight(isOpen ? "1000px" : "0px"); + }, [isOpen]); + + useEffect(() => { + if (contentRef.current && data.length > GITHUB_PAGE_LIMIT) { + contentRef.current.scrollTo({ + top: contentRef.current.scrollHeight, + behavior: "smooth", + }); + } + }, [data]); + + return ( +
+
+
{title}
+
+ {"▶"} +
+
+
+
+ {data.map((entry, index) => ( +
+
setHoveredIndex(index)} + onMouseLeave={() => setHoveredIndex(null)} + onClick={() => + window.open( + entry.html_url, + "_blank", + "noopener,noreferrer", + ) + } + > +
+ {entry.bullet && ( +
+ )} +
{entry.title}
+
+
+ {entry.created_at.split("T")[0]} +
+
+
+ ))} +
+
+ {isOpen && loadMore && data.length < total_count && ( +
+ setHoverLoadMore(true)} + onMouseLeave={() => setHoverLoadMore(false)} + onClick={loadMore} + > + Load more + +
+ )} +
+ ); +}; diff --git a/docs/community/components/Contributions.tsx b/docs/community/components/Contributions.tsx new file mode 100644 index 0000000000..39fed8bbbe --- /dev/null +++ b/docs/community/components/Contributions.tsx @@ -0,0 +1,363 @@ +import React, { useState, useEffect } from "react"; +import { Accordion } from "./Accordion"; +import { StatCard } from "./StatCard"; +import { THEME_COLORS } from "./Contributors"; +import { hexToRgb, useGithubAccessToken } from "./utils"; +import ScoreIcon from "./ScoreIcon"; +import Summary from "./Summary"; +import Hero from "./Hero"; + +export interface GitHubItem { + html_url: string; + title: string; + created_at: string; + bullet?: string; +} + +export interface StatCardProps { + title: string; + value: number; + style?: React.CSSProperties; +} + +export interface AccordionItem { + data: GitHubItem[]; + total_count: number; + state?: string; +} + +export enum BULLET_COLOR { + OPEN = "#1A7F37", + CLOSE = "#D1242F", + MERGE = "#8250DF", +} + +const initializeAccordionItem = (): AccordionItem => ({ + data: [], + total_count: 0, +}); + +const Contributions = ({ + contributor, + onBack, + darkMode, + activitySummary, + score, +}) => { + const githubAccessToken = useGithubAccessToken(); + const [commitsData, setCommitsData] = useState( + initializeAccordionItem(), + ); + const [prsData, setPrsData] = useState( + initializeAccordionItem(), + ); + const [issuesData, setIssuesData] = useState( + initializeAccordionItem(), + ); + const [openAccordion, setOpenAccordion] = useState(null); + + const [commitPage, setCommitPage] = useState(1); + const [prPage, sePrPage] = useState(1); + const [issuePage, setIssuePage] = useState(1); + + useEffect(() => { + const fetchContributorStats = async () => { + try { + await fetchCommits(commitPage); + await fetchPRs(prPage); + await fetchIssues(issuePage); + } catch (error) { + console.error("Error fetching contributor stats:", error); + } + }; + + fetchContributorStats(); + }, [contributor.login]); + + const toggleAccordion = (section: string) => { + setOpenAccordion((prev) => (prev === section ? null : section)); + }; + + const fetchCommits = async (page: number) => { + try { + const commitResponse = await fetch( + `https://api.github.com/repos/ai16z/eliza/commits?author=${contributor.login}&page=${page}`, + { + method: "GET", + headers: { + Authorization: `token ${githubAccessToken}`, + Accept: "application/vnd.github.v3+json", + }, + }, + ); + const commitData = await commitResponse.json(); + const commitItems = commitData.map((commit: any) => ({ + html_url: commit.html_url, + title: commit.commit.message, + created_at: commit.commit.author.date, + })); + const currentCommitsData = [...commitsData.data, ...commitItems]; + setCommitsData({ + data: currentCommitsData, + total_count: contributor.contributions, + }); + } catch (error) { + console.error("Error fetching commits:", error); + } + }; + + const fetchPRs = async (page: number) => { + try { + const prResponse = await fetch( + `https://api.github.com/search/issues?q=type:pr+author:${contributor.login}+repo:ai16z/eliza&page=${page}`, + { + method: "GET", + headers: { + Authorization: `token ${githubAccessToken}`, + Accept: "application/vnd.github.v3+json", + }, + }, + ); + const prData = await prResponse.json(); + const prItems = prData.items.map((pr: any) => ({ + html_url: pr.html_url, + title: pr.title, + created_at: pr.created_at, + bullet: + pr.state === "open" + ? BULLET_COLOR.OPEN + : pr.pull_request.merged_at + ? BULLET_COLOR.MERGE + : BULLET_COLOR.CLOSE, + })); + const currentPrsData = [...prsData.data, ...prItems]; + + setPrsData({ + data: currentPrsData, + total_count: prData.total_count, + }); + } catch (error) { + console.error("Error fetching PRs:", error); + } + }; + + const fetchIssues = async (page: number) => { + try { + const issueResponse = await fetch( + `https://api.github.com/search/issues?q=type:issue+author:${contributor.login}+repo:ai16z/eliza&page=${page}`, + { + method: "GET", + headers: { + Authorization: `token ${githubAccessToken}`, + Accept: "application/vnd.github.v3+json", + }, + }, + ); + const issueData = await issueResponse.json(); + const issueItems = issueData.items.map((issue: any) => ({ + html_url: issue.html_url, + title: issue.title, + created_at: issue.created_at, + bullet: + issue.state === "open" + ? BULLET_COLOR.OPEN + : BULLET_COLOR.CLOSE, + })); + const currentIssuesData = [...issuesData.data, ...issueItems]; + setIssuesData({ + data: currentIssuesData, + total_count: issueData.total_count, + }); + } catch (error) { + console.error("Error fetching issues:", error); + } + }; + + const accordionItems = [ + { + title: "Commits", + data: commitsData, + section: "commits", + loadMore: () => { + const nextPage = commitPage + 1; + fetchCommits(nextPage); + setCommitPage(nextPage); + }, + }, + { + title: "Pull Requests", + data: prsData, + section: "pullRequests", + loadMore: () => { + const nextPage = prPage + 1; + fetchPRs(nextPage); + sePrPage(nextPage); + }, + }, + { + title: "Issues", + data: issuesData, + section: "issues", + loadMore: () => { + const nextPage = issuePage + 1; + fetchIssues(nextPage); + setIssuePage(nextPage); + }, + }, + ]; + + return ( +
+
+ + + back + +
+
+ + +
+ + + +
+ {accordionItems.map((stat, index) => ( + + ))} +
+
+ {accordionItems.map((item) => ( + toggleAccordion(item.section)} + data={item.data.data} + loadMore={item.loadMore} + total_count={item.data.total_count} + primaryText={ + darkMode + ? THEME_COLORS.dark.primaryText + : THEME_COLORS.light.primaryText + } + secondaryText={ + darkMode + ? THEME_COLORS.dark.secondaryText + : THEME_COLORS.light.secondaryText + } + mainBackgroundColor={ + darkMode + ? THEME_COLORS.dark.mainBackgroundColor + : THEME_COLORS.light.mainBackgroundColor + } + /> + ))} +
+
+ ); +}; + +export default Contributions; diff --git a/docs/community/components/Contributor.tsx b/docs/community/components/Contributor.tsx new file mode 100644 index 0000000000..45efcb551e --- /dev/null +++ b/docs/community/components/Contributor.tsx @@ -0,0 +1,96 @@ +import React, { useState } from "react"; +import { ContributorProps } from "./Contributors"; +import { THEME_COLORS } from "./Contributors"; +import { hexToRgb } from "./utils"; +import ScoreIcon from "./ScoreIcon"; +import Summary from "./Summary"; +import Hero from "./Hero"; + +const ContributorCard: React.FC = ({ + contributor, + onSelect, + darkMode, + activitySummary, + score, +}) => { + const [isHovered, setIsHovered] = useState(false); + + return ( +
setIsHovered(true)} + onMouseLeave={() => setIsHovered(false)} + onClick={onSelect} + > + + + +
+ ); +}; + +export default ContributorCard; diff --git a/docs/community/components/Contributors.tsx b/docs/community/components/Contributors.tsx new file mode 100644 index 0000000000..ecbd730f68 --- /dev/null +++ b/docs/community/components/Contributors.tsx @@ -0,0 +1,233 @@ +import React, { useEffect, useState, useRef } from "react"; +import ContributorCard from "./Contributor"; +import Contributions from "./Contributions"; +import { useColorMode } from "@docusaurus/theme-common"; +import contributorsSpec from "../contributors.json"; +import { useGithubAccessToken } from "./utils"; + +export interface Contributor { + id: number; + login: string; + avatar_url: string; + html_url: string; + contributions: number; +} + +export interface ContributorProps { + contributor: Contributor; + onSelect: () => void; + darkMode: boolean; + activitySummary?: string; + score?: number; +} + +export const THEME_COLORS = { + light: { + mainBackgroundColor: "#ffffff", + secondaryBackground: "rgba(0, 0, 0, 0.05)", + primaryText: "#000000", + secondaryText: "#ffa600", + }, + dark: { + mainBackgroundColor: "#1b1b1d", + secondaryBackground: "#242526", + primaryText: "#ffffff", + secondaryText: "#add8e6", + }, +}; + +export interface ActivityDetails { + score: number; + activitySummary: string; +} + +export const GITHUB_PAGE_LIMIT = 30; // The maximum number to fetch per page from the GitHub API. + +const Contributors: React.FC = () => { + const githubAccessToken = useGithubAccessToken(); + const { colorMode } = useColorMode(); + const [selectedContributor, setSelectedContributor] = + useState(null); + const [contributors, setContributors] = useState([]); + const [error, setError] = useState(null); + const [darkMode, setDarkMode] = useState(colorMode === "dark"); + const [hasMore, setHasMore] = useState(true); + const [activitySummaries, setActivitySummaries] = useState< + Map + >(new Map()); + + const observerRef = useRef(null); + const pageRef = useRef(1); + const loadingRef = useRef(true); + + useEffect(() => { + setDarkMode(colorMode === "dark"); + }, [colorMode]); + + const fetchContributors = async (page: number) => { + loadingRef.current = true; + try { + const response = await fetch( + `https://api.github.com/repos/ai16z/eliza/contributors?per_page=${GITHUB_PAGE_LIMIT}&page=${page}`, + { + method: "GET", + headers: { + Authorization: `token ${githubAccessToken}`, + Accept: "application/vnd.github.v3+json", + }, + }, + ); + if (!response.ok) { + throw new Error( + `Error fetching contributors: ${response.statusText}`, + ); + } + const data: Contributor[] = await response.json(); + if (data.length === 0) { + setHasMore(false); + return; + } + const currentContributors = [...contributors, ...data]; + + setContributors(currentContributors); + } catch (err) { + setError(err instanceof Error ? err.message : "Unknown error"); + } finally { + loadingRef.current = false; + } + }; + + useEffect(() => { + const fetchActivitySummaries = async () => { + try { + const response = await fetch( + "https://ai16z.github.io/data/contributors.json", + ); + if (!response.ok) { + throw new Error( + `Error fetching activity summaries: ${response.statusText}`, + ); + } + const specs = await response.json(); + + const currentActivitySummaries = new Map< + string, + ActivityDetails + >(); + specs.forEach( + (spec: { + contributor: string; + score: number; + summary: string; + }) => { + currentActivitySummaries.set(spec.contributor, { + score: spec.score, + activitySummary: spec.summary, + }); + }, + ); + setActivitySummaries(currentActivitySummaries); + } catch (err) { + console.log("Unknown error while fetching summaries"); + } + }; + + fetchActivitySummaries(); + fetchContributors(pageRef.current); + }, []); + + useEffect(() => { + const observer = new IntersectionObserver( + (entries) => { + if ( + entries[0].isIntersecting && + !loadingRef.current && + hasMore + ) { + loadingRef.current = true; + pageRef.current++; + fetchContributors(pageRef.current); + } + }, + { threshold: 1.0 }, + ); + + if (observerRef.current) { + observer.observe(observerRef.current); + } + + return () => { + if (observerRef.current) { + observer.unobserve(observerRef.current); + } + }; + }, [contributors, hasMore, selectedContributor]); + + if (error) { + return
Error: {error}
; + } + + if (!contributors.length) { + return
Loading...
; + } + + return ( +
+ {selectedContributor ? ( + setSelectedContributor(null)} + darkMode={darkMode} + activitySummary={ + activitySummaries.get(selectedContributor.login) + ?.activitySummary + } + score={ + activitySummaries.get(selectedContributor.login)?.score + } + /> + ) : ( + <> + {contributors.map((contributor) => ( + { + setSelectedContributor(contributor); + }} + darkMode={darkMode} + activitySummary={ + activitySummaries.get(contributor.login) + ?.activitySummary + } + score={ + activitySummaries.get(contributor.login)?.score + } + /> + ))} +
+ {hasMore &&
Loading more...
} + + )} +
+ ); +}; + +export default Contributors; diff --git a/docs/community/components/Hero.tsx b/docs/community/components/Hero.tsx new file mode 100644 index 0000000000..4575fd4080 --- /dev/null +++ b/docs/community/components/Hero.tsx @@ -0,0 +1,47 @@ +import React from "react"; + +export default function Hero({ + contributor, + secondaryText, + profilePictureSize, +}) { + return ( +
+ {`${contributor.login}'s +
+
+ {contributor.login} +
+ { +
+ {contributor.contributions} contributions +
+ } +
+
+ ); +} diff --git a/docs/community/components/ScoreIcon.tsx b/docs/community/components/ScoreIcon.tsx new file mode 100644 index 0000000000..fcf5ae325c --- /dev/null +++ b/docs/community/components/ScoreIcon.tsx @@ -0,0 +1,32 @@ +import React from "react"; + +export default function ScoreIcon({ style, iconColor, iconSize, score }) { + function Flash({ size, fill }) { + return ( + + + + ); + } + + return ( +
+ +
{typeof score === "number" ? score : "NULL"}
+
+ ); +} diff --git a/docs/community/components/StatCard.tsx b/docs/community/components/StatCard.tsx new file mode 100644 index 0000000000..d812de946f --- /dev/null +++ b/docs/community/components/StatCard.tsx @@ -0,0 +1,22 @@ +import React from "react"; +import { StatCardProps } from "./Contributions"; + +export const StatCard: React.FC = ({ title, value, style }) => { + return ( +
+
{title}
+
{value}
+
+ ); +}; diff --git a/docs/community/components/Summary.tsx b/docs/community/components/Summary.tsx new file mode 100644 index 0000000000..b9a0a90636 --- /dev/null +++ b/docs/community/components/Summary.tsx @@ -0,0 +1,13 @@ +import React from "react"; + +export default function Summary({ summary, style }) { + return ( +
+ {summary || "No summary available"} +
+ ); +} diff --git a/docs/community/components/utils.tsx b/docs/community/components/utils.tsx new file mode 100644 index 0000000000..bd660f5959 --- /dev/null +++ b/docs/community/components/utils.tsx @@ -0,0 +1,15 @@ +import useDocusaurusContext from "@docusaurus/useDocusaurusContext"; + +export function hexToRgb(hex: string) { + hex = hex.replace("#", ""); + const bigint = parseInt(hex, 16); + const r = (bigint >> 16) & 255; + const g = (bigint >> 8) & 255; + const b = bigint & 255; + return `${r}, ${g}, ${b}`; +} + +export function useGithubAccessToken() { + const { siteConfig } = useDocusaurusContext(); + return siteConfig.customFields.GITHUB_ACCESS_TOKEN; +} diff --git a/docs/community/profiles.mdx b/docs/community/profiles.mdx index 28224acecd..5135aede38 100644 --- a/docs/community/profiles.mdx +++ b/docs/community/profiles.mdx @@ -1,17 +1,10 @@ --- -title: GitHub Contributors +title: GitHub Contributors description: GitHub contributors to our project --- -# GitHub Contributors +import Contributors from "./components/Contributors"; -This is a quick and dirty implementation of profiles that are programmatically generated from github data from `ai16z/eliza` repo. I'm looking for some help to integrate into Docusaurus as react components. See the code for generating profiles here: https://github.com/ai16z/ai16z.github.io +# GitHub Contributors -