Auth0 simulator
Table of Contents
Please read the main README for more background on simulacrum.
The auth0 simulator has been initially written to mimic the responses of a real auth0 server that is called from auth0 client libraries like auth0/react and auth0-spa-js that use the OpenID authorization code flow.
If this does not meet your needs then please create a github issue to start a conversation about adding new OpenID flows.
Quick Start
Graphql
Let's start our server.
npm install
cd ./packages/auth0
PORT=4000 npm run start
Open a browser at http://localhost:4000.
This will open a graphql graphiql editor.
Enter the following mutation:
mutation CreateSimulation {
createSimulation(simulator: "auth0",
options: {
options:{
audience: "[your audience]",
scope: "[your scope]",
clientId: "[your client-id]"
},
services:{
auth0:{
port: 4400
}
}
}) {
id
status
services {
url
name
}
}
}
This mutation creates your first simulation, and runs it in memory. Every time you start the server, you will need to apply these mutations. This can also be done programmatically which will be your likely interface while writing tests.
.
Use the values returned from the query to update your configuration in the client application that calls the auth0 endpoints as shown below. This will point your app at the simulation instead of the Auth0 endpoint.
{
"domain": "localhost:4400",
"clientId": "00000000000000000000000000000000",
"audience": "https://your-audience/"
}
Create a fake user whose credentials can be used for authentication with this query.
mutation CreatePerson {
given(a: "person", simulation: "6fbe024f-2316-4265-a6e8-d65a837e308a")
}
The fake user can now be used on your app. Use the email
and password
fields as login credentials.
You now have a running auth0 server!
Code
An auth0 simulator can be created using the @simulacrum/client
package. This is how you would apply the mutations programmatically.
The following examples are written in Typescript, but using Typescript is not a requirement. The Auth0 simulator creates a server with a graphql interface. This means that your interactions with the server can be written in any language or framework that can communicate over http / graphql.
import { main } from "effection";
import { createSimulationServer, Server } from "@simulacrum/server";
import { auth0 } from ".";
import { createClient } from "@simulacrum/client";
const port = Number(process.env.PORT) ?? 4000;
main(function* () {
let server: Server = yield createSimulationServer({
seed: 1,
port,
simulators: { auth0 },
});
let url = `http://localhost:${server.address.port}`;
console.log(`simulation server running at ${url}`);
let client = createClient(url);
let simulation = yield client.createSimulation("auth0", {
options: {
audience: "[your audience]",
scope: "[your scope]",
clientId: "[your client-id]",
},
services: {
auth0: {
port: 4400,
},
},
});
console.log(`auth0 service running at ${simulation.services[0].url}`);
let person = yield client.given(simulation, "person");
console.log(`store populated with user`);
console.log(
`username = ${person.data.email} password = ${person.data.password}`
);
yield;
});
The client
is also expected to be run in many different contexts, and, as such, supports async/await as well.
import { main } from "effection";
import { createSimulationServer } from "@simulacrum/server";
import { auth0 } from "@simulacrum/auth0-simulator";
import { createClient } from "@simulacrum/client";
const port = Number(process.env.PORT) || 4000;
main(function* startServer() {
let server = yield createSimulationServer({
port,
simulators: { auth0 },
});
let url = `http://localhost:${server.address.port}`;
console.log(`simulation server running at ${url}`);
yield setupClient({ url });
});
async function setupClient({ url }) {
let client = createClient(url);
let simulation = await client.createSimulation("auth0", {
options: {
audience: "https://your-audience/",
scope: "openid profile read:shows",
clientId: "YOUR_AUTH0_CLIENT_ID",
},
services: {
auth0: {
port: 4400,
},
},
});
console.log(`auth0 service running at ${simulation.services[0].url}`);
let person = await client.given(simulation, "person");
console.log(`store populated with user`);
console.log(
`username = ${person.data.email} password = ${person.data.password}`
);
}
Configuration
Both the graphql createSimulation
mutation and the @simulacrum/client
take an optional options
and services
object.
let simulation = yield client.createSimulation("auth0", {
options: {
audience: "[your audience]",
scope: "[your scope]",
clientId: "[your client-id]",
rulesDirectory: "test/rules",
},
services: {
auth0: {
port: 4400,
},
},
});
Options
The options
field supports the auth0 configuration fields. The option fields should match the fields in the client application that is calling the auth0 server.
An optional rulesDirectory
field can specify a directory of auth0 rules code files, more on this below.
Services
The services
object configures simulators to start on specific ports.
let simulation = yield client.createSimulation("auth0", {
options: {
services: {
auth0: {
port: 4400,
},
},
},
});
Rules
It is possible to run auth0 rules if the compiled code files are on disk and all located in the same directory.
Set the rulesDirectory
of the options field to a path relative to your current working directory.
For example, a sample rules directory is in the auth0 package for testing.
If we want to run these rules files then we would add the rulesDirectory
field to the options object.
let simulation = yield client.createSimulation(url, {
options: {
rulesDirectory: "test/rules",
},
});
Endpoints
The following endpoints have been assigned handlers:
/authorize
/login
/u/login
/usernamepassword/login
/login/callback
/oauth/token
/v2/logout
/.well-known/jwks.json
/.well-known/openid-configuration