An AI agent acted across two companies. Whose audit log knows which human? A developer built Crumb, a system that creates tamper-evident audit logs binding individual humans to AI agent tool calls across different identity providers. The system solves the cross-company delegation problem by stapling upstream tokens into new ones, preserving cryptographic proof of the original human identity. This allows auditors to verify the human behind an agent's action without trusting the operator. Here is a setup that is going to be normal soon, if it isn't already. Alice logs into her company's tools through their identity provider. She points an agent at a task. That agent hands part of the work to a sub-agent, and the sub-agent calls a tool that lives in a partner company's system, behind a different identity provider. The tool does something it shouldn't. An auditor pulls the record. Whose log knows it was alice? Not the agent's. The agent is a process; it can claim to be anyone. Not the model's either, which reads whatever it was handed and has no idea which human is behind the session. The honest answer in most deployments today is that the partner's system can prove a bot called it, and can prove which company's bot , and then the trail goes cold. The person who actually directed the action dissolves into "some agent at the vendor." I have been building Crumb https://crumb.alexlaguardia.dev to refuse that outcome: a tamper-evident record that binds the individual human behind an agent's tool call, verifiable by someone who does not have to trust whoever ran the agent. Within a single identity provider, that chain was already working. This post is about the part that wasn't, and why it took longer than I expected. When the whole chain lives under one identity provider, delegation has a clean answer, and it is a real standard. RFC 8693 token exchange lets you mint a token that carries two identities at once: the human as the sub , and the agent acting for them as a nested act claim. Add a hop and you nest again. The human stays at the root the whole way down. { "iss": "https://idp-a.local", "sub": "alice", "act": { "sub": "researcher", "act": { "sub": "planner" } }, "aud": "read record" } One provider signs that token. A resource server verifies it against that provider's public key, walks the act chain back to alice, and it is done. No shared secret, no trusting the gateway that minted it. I covered that build in an earlier post. It holds up. The catch is in the assumption hiding under "one provider." Real delegation does not stay inside one company. The interesting, dangerous case is the one that crosses. So planner , holding a token IdP A signed, needs the call into B's domain to carry a token B will honor. The textbook move is another RFC 8693 exchange, this time against B. You hand B the token A issued, and B mints you a fresh one. And right there is the problem, sitting in plain sight in the spec. When B does that exchange, it mints a token signed only by B and drops A's signature on the floor. The new token says sub: alice because B copied it across, but the cryptographic proof that A authenticated alice is gone. Downstream, all you hold is B's word: "A told me it was alice." For most systems that is fine, because most systems were already trusting B. But Crumb's entire reason to exist is to let an auditor verify without trusting the operator. A cross-issuer hop that resolves to "trust B" puts the trust-me point right back in the middle of the chain I was trying to make checkable. It is the one thing I am not allowed to wave away. The fix I landed on is to stop throwing the upstream token away. When B exchanges A's token, two things happen. First, B verifies A's token against A's public key. B can only do that if it federates with A, so A has to be in B's trust set. That is a real relationship and I will come back to how honest it is. Second, instead of discarding A's token, B staples it into the one it mints: the exact inner JWS rides along in a prv claim, its SHA-256 in psh , and the inner issuer in pis . { "iss": "https://idp-b.local", "sub": "alice", "act": { "sub": "researcher", "act": { "sub": "planner" } }, "aud": "read record", "prv": "