Building an Alien Language from Scratch with LangChain A developer built a system where 100 AI agents, each with unique personalities, evolved a shared symbolic vocabulary through trade negotiations using LangChain.js pipelines, with no pre-programmed symbol definitions or mock data. The project uses three different LangChain patterns to handle agent complexity, including a custom `LenientJsonParser` that extends LangChain's `JsonOutputParser` to handle real-world LLM quirks like markdown code fences and empty content fields. The system supports instant provider swapping between OpenAI, Featherless.ai, and LM Studio by changing a single `baseURL` configuration. How 100 AI agents evolved a shared symbolic vocabulary through trade — with no pre-programmed definitions, no mock data, and no shortcuts. Most emergent communication demos cheat. They hardcode symbol meanings, or use mock LLM responses, or simulate the entire thing with random number generators. The result is a demo that looks impressive but teaches you nothing about how actual language models behave when they need to coordinate. I wanted to build something real: two civilizations of AI agents, each with 50 unique personalities, negotiating trades using abstract symbols — where every single message and decision flows through a real LLM via LangChain SDK pipelines . No mock data. No hardcoded symbol mappings. Just pure reinforcement: successful trades reinforce a symbol's meaning, failed trades weaken it. Over hundreds of rounds, a shared vocabulary emerges. LangChain.js v1.4 is the backbone of this project. I deliberately used three different LangChain patterns to show how the SDK can handle different levels of agent complexity: js const chain = ChatPromptTemplate .fromMessages "system", prompt , "human", "{input}" .pipe model .pipe parser ; Every symbol an agent sends is generated through this pipeline. The ChatPromptTemplate injects the agent's personality, observations, recent successes/failures, and conversation history. The model returns structured JSON {"message": "🟢⚡🔺"} , which the JsonOutputParser validates. The trickiest part was the {{ escaping — LangChain uses {} for template variables, but I needed literal {} in the JSON example. The double-brace escape {{"message": "..."}} resolves this cleanly. Same structure, different prompt — but this one returns a boolean decision. {"accept": true} or {"accept": false} . The agent evaluates whether the proposed resource exchange is beneficial based on its observations and personality. This is where LangChain really shines. Using createAgent from the LangGraph-based SDK, I built a full ReAct agent that can: DynamicTool DynamicTool The agent literally thinks, uses tools, and then decides — all within LangChain's agent loop. The moment you switch from mock to real LLMs, everything gets interesting. Models wrap JSON in markdown code fences. They include chain-of-thought reasoning. They run out of tokens mid-response. They return empty content and put their thoughts in vendor-specific fields like reasoning content . I built a LenientJsonParser that extends LangChain's JsonOutputParser to handle all of these cases: class LenientJsonParser extends JsonOutputParser { async parse text: string : Promise