Deployment
A Transiter deployment has two pieces:
-
A running Postgres instance that has the PostGIS extension available. If you've installed Postgres using your operating system's package manager (like
apt
), or if you're using a managed Postgres offering from one of the Cloud providers, PostGIS is probably already installed. Even easier is to use the Docker imagepostgis/postgis:14-3.4
. For other options, consult the PostGIS documentation. -
The Transiter binary. This is available in a Docker image, or can be built by running
go build .
in the Transiter repository, or by downloading a prebuilt binary from the GitHub releases page.
Because the deployment model is so simple, there are many ways to deploy Transiter.
We recommend deploying Transiter using the Transiter Docker image.
This Docker image is built on every push to mainline and
stored in Docker Hub.
The latest build always has tag jamespfennell/transiter:latest
.
Each build on mainline gets its own specific tag too,
in case you want to pin to a specific version.
The tags have one of two forms:
-
For releases of version vA.B.C, the tag is
jamespfennell/transiter:vA.B.C
. The available releases are listed on the GitHub releases page. -
For non-release builds on mainline of version vA.B.C, the tag is
jamespfennell/transiter:vA.B.C-beta.buildN
, where N is a GitHub build number. The easiest way to find a valid tag is to browse the tags in Docker Hub.
This is an example Docker compose configuration which runs the Transiter Docker image alongside a Postgres Docker image:
version: '3.5'
services:
transiter:
image: jamespfennell/transiter:1.0.0
ports:
- "127.0.0.1:8080:8080"
- "127.0.0.1:8081:8081"
- "127.0.0.1:8082:8082"
- "127.0.0.1:8083:8083"
restart: always
command:
- server
- --log-level
- info
- --postgres-connection-string
- "postgres://transiter:transiter@postgres:5432/transiter"
postgres:
image: postgis/postgis:14-3.4
environment:
- POSTGRES_USER=transiter
- POSTGRES_PASSWORD=transiter
- POSTGRES_DB=transiter
volumes:
# Persist the Postgres data outside the container to that it persists across restarts.
- "${HOME}/transiter-data:/var/lib/postgresql/data"
restart: always
The Postgres instance doesn't need to be on the same machine. You can even used a managed Postgres service like Google's CloudSQL.
Reverse proxy
Generally Transiter should be run behind a reverse proxy like Caddy or Nginx, for the usual reverse proxy reasons. This is a sample Caddyfile configuration for Caddy:
demo.transiter.dev {
encode gzip
reverse_proxy 127.0.0.1:8080 {
header_up X-Transiter-Host "https://demo.transiter.dev"
}
}
In this case requests to demo.transiter.dev
are reverse proxied to localhost:8080
.
This is where the Transiter public HTTP API is listening if you used the Docker compose
configuration above.
Alternatively, you can use a Docker compose configuration such as below:
version: '3.5'
services:
transiter:
image: jamespfennell/transiter:latest
ports:
- "127.0.0.1:8080:8080"
- "127.0.0.1:8081:8081"
- "127.0.0.1:8082:8082"
- "127.0.0.1:8083:8083"
restart: always
command:
- server
- --log-level
- info
- --postgres-connection-string
- "postgres://transiter:transiter@postgres:5432/transiter"
caddy:
image: caddy:2.8.4
ports:
- "127.0.0.1:8090:8090"
command:
- caddy
- reverse-proxy
- --from
- :8090
- --to
- transiter:8080
- --header-up
- "X-Transiter-Host: https://demo.transiter.dev"
postgres:
image: postgis/postgis:14-3.4
environment:
- POSTGRES_USER=transiter
- POSTGRES_PASSWORD=transiter
- POSTGRES_DB=transiter
restart: always
(Optional) Configuring CORS
If you plan to make requests to Transiter from a web browser, you may run into CORS
issues if you do not run Transiter on the same origin as your web application. Since the Transiter public API is generally accessible using
"Simple requests", you can simply configure your reverse proxy to add the
Access-Control-Allow-Origin
header to all responses from Transiter and avoid handling CORS "preflight" requests.
Note
For local web application development, it is likely sufficient to use the built-in proxy functionality of your bundler/framework. For example:
- Create React App's
proxy
option - Vite's
server.proxy
option - Next.js's
rewrites
configuration
Below is a simple Caddy configuration that adds the Access-Control-Allow-Origin
header to all responses from Transiter:
Warning
The below examples allow any origin to access your Transiter host. In most cases you should specify a more
specific Access-Control-Allow-Origin
header to properly limit access.
demo.transiter.dev {
encode gzip
reverse_proxy 127.0.0.1:8080 {
header_up X-Transiter-Host "https://demo.transiter.dev"
header_down Access-Control-Allow-Origin "*"
}
}
And here is the equivalent Caddy setup within a Docker compose configuration:
caddy:
image: caddy:2.8.4
ports:
- "127.0.0.1:8090:8090"
command:
- caddy
- reverse-proxy
- --from
- :8090
- --to
- transiter:8080
- --header-up
- "X-Transiter-Host: https://demo.transiter.dev"
- --header-down
- "Access-Control-Allow-Origin: *"
(Optional) Setting the Transiter host
The reverse proxy configuration above contains a X-Transiter-Host
header instruction.
This tells Caddy that when a HTTP request is forwarded to Transiter, the X-Transiter-Host
header should be added with value https://demo.transiter.dev
.
The point of this optional header is to support Transiter's version of "HATEOAS".
When this header is set, every resource that is returned will have the full URL of the resource in the response.
For example, when listing systems on the demo site,
the NYC subway resource contains the exact URL https://demo.transiter.dev/systems/us-ny-subway
for the resource.
In general Transiter knows nothing about the base URL, so the X-Transiter-Host
provides it.
If the header is missing, Transiter simply skips returning URLs.
Admin APIs
In general you will only want to make the public HTTP API (default port 8080) and/or public gRPC API (default port 8081) available to the internet. The admin APIs (on ports 8082 and 8083) should be hidden so that internet users can't e.g. delete all your transit systems. With the Docker compose configuration above, these APIs are still available locally on the machine so you can still administer the Transiter instance by SSHing into the machine.
Sometimes you may want to administer a remote Transiter instance from your personal computer. You can use SSH port forwarding to do this. If your admin gRPC API is listing on port 8083 (as above), run the following command on your personal computer:
ssh -N -L8083:localhost:8083 $HOST_RUNNING_TRANSITER
Now port 8083 on your personal computer is linked to the Transiter admin API on the remote machine. This means you can run something like:
transiter list
on your personal machine and it will work. You can also install/delete transit systems this way, etc.
Monitoring
After deploying Transiter you may be want to monitor it.