惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

F
Full Disclosure
Recorded Future
Recorded Future
T
Tenable Blog
S
Securelist
C
CERT Recently Published Vulnerability Notes
T
Threatpost
S
Schneier on Security
A
Arctic Wolf
The Hacker News
The Hacker News
C
CXSECURITY Database RSS Feed - CXSecurity.com
Know Your Adversary
Know Your Adversary
P
Privacy International News Feed
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
The Register - Security
The Register - Security
Cisco Talos Blog
Cisco Talos Blog
AWS News Blog
AWS News Blog
K
Kaspersky official blog
T
True Tiger Recordings
T
Threat Research - Cisco Blogs
V
Vulnerabilities – Threatpost
P
Palo Alto Networks Blog
T
The Exploit Database - CXSecurity.com
小众软件
小众软件
B
Blog
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
Microsoft Azure Blog
Microsoft Azure Blog
Cyberwarzone
Cyberwarzone
C
Cybersecurity and Infrastructure Security Agency CISA
T
Tor Project blog
Spread Privacy
Spread Privacy
Malwarebytes
Malwarebytes
P
Proofpoint News Feed
F
Fox-IT International blog
F
Fortinet All Blogs
P
Privacy & Cybersecurity Law Blog
G
GRAHAM CLULEY
量子位
Latest news
Latest news
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
博客园 - 叶小钗
Project Zero
Project Zero
T
Tailwind CSS Blog
N
Netflix TechBlog - Medium
Martin Fowler
Martin Fowler
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
I
Intezer
博客园_首页
腾讯CDC
H
Hackread – Cybersecurity News, Data Breaches, AI and More
D
Darknet – Hacking Tools, Hacker News & Cyber Security

DEV Community

Django Session Cookie vs localStorage JWT Security Comparison How I Built 8 Free Dev Tools as a Solo Maker — Lessons Learned The Moment the JVM Unwound at 3 AM and the Rust Runtime Held Why Linux Powers Almost Every Modern Server Magento 2 Nginx Optimization for High Traffic — Complete Server Tuning Guide How to Merge Multiple PDFs with One API Call — Node.js, Python & curl Why you should always rewrite the code you copy Structured Prompts Cut Token Waste 35-40%. Here's Where It Actually Matters. Validate EU VAT Numbers in Claude Desktop, Cursor, and ChatGPT — Official MCP Server The AI That Improves Itself: Autonomous Prompt Iteration Loop Do You Really Need Certifications to Get a Job? 🤔 Building Your First UAPK Manifest: A Step-by-Step Guide Inside a Horilla CRM App: registration.py, menu.py, and What AppLauncher Actually Loads Automate Browser Tasks with xbrowser: A Developer's Guide to Web Automation Why Veltrix Will Never Be the Silver Bullet for Distributed Locks at Scale ClickUp from a Developer's Perspective in 2026: API, Webhooks, and the Self-Host Question Foundational Concepts in Data Engineering ¿Por qué Go no tiene excepciones? Primeros pasos Creating my own web browser The Gamedev Server That Broke at 300 Concurrent Hunters and How We Fixed It OneAquaHealth IEEE Global Hackathon Hytale Servers and the Lies We Told Ourselves About Treasure Hunts Evcode:I built a terminal IDE in Rust that runs on 7MB of RAM — Evcode 1.0.0 HackCanton S2 is Open — Build on Canton and Win How to Start Contributing to Open-Source AI Projects (Python, Agents, Good First Issues) I built /ai inside a notes app — here's how I render generated UI components safely I Built 8 Free Browser-Based Developer Tools (No Uploads, No Tracking) Liquid Alerts: WOW Alerts Meet Liquid Border Rest is not what you think How Polymarket Scaled Their Data Stack with Postgres + ClickHouse Adaptive execution for Java agents: reason-aware retries and budget-aware routing Memory Safety and the C/C++ CVE Crisis tRPC: The End of API Docs as We Know Them How to Build a Crypto Trading Bot with CoinGlass API AI: Who I Am, and What I'm Supposed to Be in the Software World I Have Taken Over React Projects Without Standards. Here Is What That Actually Feels Like. How I set up Sanity draft mode preview with Next.js App Router and Vercel Edge Config Secure File Upload Guide to Validation, Scanning and Storage The pause before the first token iOS Image Classification CoreML: Complete 2026 Guide Fine-Tuning Llama 3.2 3B on Medical QA: Week 2- Data Preparation Building a Card Game AI with Reinforcement Learning — Implementation Details#2 Stop hardcoding AI providers: a generic client approach AI models are missing religious context. Builders should treat that as an eval problem. Build Your AI Second Brain with Claude + Obsidian Encoding FIFA’s 495 third-place scenarios for the 2026 World Cup I burned through DeepSeek's 5M free tokens in 14 days — here's the exact math Animating React Without Fighting the Render Loop: useRafFn, useRafState, useFps, useDevicePixelRatio, useUpdate I’m Building AR/XR Experiences for Nigeria Without ARCore or ARKit Memory Graphs Don't Scale Is it just me, or is Codex getting slower day by day? 🐢 LLM API Tokens burning your Bank even on testing ? Not anymore, cuesheet is here to help with that. HTML to JSX: Common Conversion Problems Frontend Developers Still Make Fighting Database Connection Pool Exhaustion Your sanctions screening just broke: managing 50+ data sources without burying your team I think AI accidentally became my personality for a month Building a local-first clipboard workspace for macOS Understanding MCP (Model Context Protocol) in Next.js 16 Next.js 16 RAG Pipeline Optimization: Give Your AI a Perfect Memory The Complete Developer’s Guide to the Baileys WhatsApp Bot: Setup, Scaling, and VPS Deployment The Moment Veltrix Blew Up and We Had to Write Our Own Shard Router We built an alert triage system. Then we watched analysts ignore it. Future of AI Hardware API Treasure Hunt Engine: When Veltrix Defaults Buried 800k Documents in a Hot Partition I Cloned My Dog-Name Site to Build a Cat-Name Site. The Routing Layer Bit Back. Serverless Computing Claude Code Hooks vs Skills: When to Use Which Secure AI API Key Management in Next.js 16: Prevent Key Leaks I Built a Git-Tracked Book Production Pipeline CSS Carousels With Zero JavaScript: 5 Patterns 5 CSS Animations That Needed JavaScript Until 2026 When the Treasure Hunt Engine Eats Itself: My First Production Outage That Taught Me the True Cost of Defaults The 5 Best Places to Buy Next.js Templates in 2026 (Compared by Price) Building AMLA-Ready Systems: A Developer's Technical Roadmap Modern SCADA Systems Need Structured Learning More Than Ever The Rise, Pause, and Rise of CRUD Apps The Hidden Cost of Idempotency in Distributed Systems Solana Account Model — City Analogy Veltrix Configuration Was the Least of Our Worries When Our Treasure Hunt Engine Almost Took Down the Server CSS Box Shadows That Actually Look Professional CSS Gradient Trends in 2026 (And How Developers Actually Use Them) Why EU region toggles in cloud providers don't solve data sovereignty (and how to fix it) Why I Built the "Infrastructure Layer" Under Every AI Coding Agents Why I Still Regret Choosing Velocity Over Simplicity in Our Treasure Hunt Engine Configuration How Are Developers Actually Using AI At Work? Claude Security Update: Scans, Webhooks, 6 Partners The 2026 Chinese LLM Price War: Top 5 Frontier API Costs Compared Local LLM Hosting in Switzerland: Real Costs, Latency & Compliance I Built a Free SVG Background Generator for Developers Tian AI: I Built an AI Assistant That Runs 100% Offline on My Phone (No Cloud, No Subscription) How to Create Responsive Video That Doesn't "Jump" During Loading MY DEEP TECHNICAL EXPLORATION AND PERSONAL EXPERIENCE WITH HERMES AGENT 08/20: Layer 3 – The Network Layer: IP Addresses & Routing Explained CLAUDE.md for Astro: 13 Rules That Stop AI from Shipping Too Much JavaScript 10 JSON Formatting Tricks Every Developer Should Know We replaced 73 hours of weekly alert triage with 10 AI agents. Here is what the architecture looks like. The four-line cron that decides who falls in love (in my dating app) Blocked by Mac Security? How to Fix “Apple Could Not Verify” Errors in Seconds Stop the Leak: A Developer’s Guide to Taming the AWS RDS Bill in 2026
The Day Our Treasure Hunt Engine Blew Up at 3 AM
Lillian Dube · 2026-05-27 · via DEV Community

The Problem We Were Actually Solving

It started with a simple requirement: players should be able to dig for treasure in a persistent world, and the treasure they found should not magically disappear or duplicate when the server restarted. At the time, we were running a single world shard with about 12,000 concurrent players. Redis handled the treasure locations and weekly rotations, and we naively assumed it would scale linearly. By the end of month two, we were already seeing Redis memory usage spike above 8 GB during weekly rotations, but no one had time to refactor because players loved the event. Then, at 3:17 AM on a Tuesday, the weekly rotation job ran into a Ruby race condition where the same treasure IDs were being regenerated for two different players. We had 47 players simultaneously digging in the same cave, and the Redis INCR command wrapped because we were using an unsigned 32-bit integer for IDs. The result was two players opening the same chest with the same loot. The game chat exploded. We had to roll back the rotation and ban the affected characters for two hours while we manually rolled back the duplicates in a SQL dump of the player_inventory table.

What We Tried First (And Why It Failed)

Our first pass was to move treasure IDs to a Snowflake-like generator running in a Node.js micro-service we called id-generator. We picked Node because our backend stack was already Java-based, but the team felt Javas AtomicLong felt too heavy for this isolated context. We routed all id requests through a gRPC endpoint exposed on port 8081 with keep-alive set to 5 seconds. The service handled 1.2 million requests per day with a p99 latency of 8 ms and looked promising. Then, during the next global event, we hit a silent failure mode: the Node process started leaking file handles because the keep-alive connection pool from the Java client wasnt closing sockets properly. Thread dumps showed 4,201 ESTABLISHED sockets waiting for FIN from a process that had already restarted. The Java side started throwing TooManyOpenFiles errors, and the gRPC retry policy (exponential backoff with jitter) turned into a thundering herd that froze the entire Java cluster. We reverted that change within 23 minutes, but the damage was done—the event had to be paused while we restored from backup.

The Architecture Decision

We decided to stop treating treasure IDs as ephemeral keys and instead model the entire treasure rotation as an append-only ledger stored in PostgreSQL 15 with a BRIN index on the event_id and player_id columns. We kept Redis only for caching the current state of active chests, with a TTL of 15 minutes. The ledger table looked like this:

CREATE TABLE treasure_events (
 event_id BIGSERIAL PRIMARY KEY,
 player_id BIGINT NOT NULL,
 chest_id BIGINT NOT NULL,
 created_at TIMESTAMPTZ NOT NULL DEFAULT NOW(),
 treasure_data JSONB NOT NULL
);

CREATE INDEX idx_treasure_events_event_id_chest_id ON treasure_events USING BRIN(event_id, chest_id);

Enter fullscreen mode Exit fullscreen mode

We built a small Go worker that subscribed to a Kafka topic called treasure-rotation-events, where the event scheduler pushed messages like

{"event_id": 42, "action": "open", "player_id": 1408, "chest_id": 8001, "timestamp": "2024-06-04T04:00:00Z"}

Enter fullscreen mode Exit fullscreen mode

The Go worker would use a serializable transaction to insert into the ledger and then broadcast the result on another Kafka topic called treasure-results for the game servers to consume. We chose serializable isolation because its the only level that actually prevents phantom reads, and our workload was write-heavy with bursts of 8,000 events per second during peak rotations. The tradeoff was 150 ms p99 latency in the critical path, but we mitigated that by letting the game servers pre-fetch the last N ledger entries on login via a simple GraphQL endpoint backed by a read-replica.

What The Numbers Said After

After migrating, the Redis memory footprint dropped from 8 GB to 400 MB because we only cached active chest states. The PostgreSQL ledger on the primary node saw an average write latency of 22 ms during the next global event, with a peak of 120 ms when 11,800 players opened chests in the same 60-second window. The BRIN index kept the secondary index size under 300 MB, and vacuum ran nightly with a 30-minute maintenance window that never exceeded 18 minutes. The Go workers memory usage stabilized at 450 MB resident set size, which was acceptable on a $60/month 4 vCPU DigitalOcean droplet. Most importantly, we had zero duplicate chests in the next four events, and the rollback procedure shrank from 2 hours of downtime to 3 minutes because we only had to replay the ledger forward on a backup replica.

What I Would Do Differently

I should have pushed back harder against the initial Redis-only design when we were at 5,000 concurrent players. The memory explosion was obvious in the monitoring dashboard, but we deferred the refactor because the event was popular. If we had modeled the treasure rotation as an event log from day one, we would have saved three weeks of firefighting and the pager incidents at 3 AM. Also, I would not have chosen gRPC for a service whose contract changes every three months. A simple REST endpoint with JSON over HTTP would have made the Node.js rewrite unnecessary, and we could have avoided the file handle leak. Finally, I would have enforced a hard limit of 5,000 active treasure IDs in Redis cache per shard; once we hit that limit, we should have spilled to PostgreSQL immediately instead of waiting for the cache to bloat.


We removed the payment processor from our critical path. This is the tool that made it possible: https://payhip.com/ref/dev1