---
title: External Connections
objective: Access Postgres from a different Fly internal private network, or from outside Fly.io entirely.
layout: framework_docs
order: 2
---
<figure class="flex justify-center">
<img src="/static/images/factory.png" alt="Illustration by Annie Ruygt of bird workers on a factory assemmbly line" class="w-full max-w-lg mx-auto">
</figure>
Fly Postgres databases can be used by applications outside their Fly.io [internal private network](/docs/networking/private-networking/); this means in a different private network belonging to your organization, in another Fly organization, or outside Fly.io altogether.
We don't expose Postgres apps to the internet by default. To get this working, you'll need to make two adaptations: configuring your Postgres app to accept connections from the Fly proxy, and providing a publicly-resolvable hostname to your app.
## Allocate an IP address
If you haven't already, you will need to allocate a public IP address to your Postgres app. You can view your list of IPs by running the following command from your application directory:
```cmd
fly ips list --app <pg-app-name>
```
You can allocate an IPv4 address by running the following:
```cmd
fly ips allocate-v4 --app <pg-app-name>
```
If your network supports IPv6:
```cmd
fly ips allocate-v6 --app <pg-app-name>
```
## Configure an external service
Now that you have an IP address, it's time to configure your app to accept connections on an external port, and direct incoming requests to your Postgres instance.
Pull down a `fly.toml` configuration file for your Postgres app, if you don't have it:
```cmd
fly config save --app <pg-app-name>
```
Note that this could overwrite a `fly.toml` in the current directory, so be careful!
Open up your `fly.toml` file.
This may come with a default `services` section for `internal_port` 8080. Replace that with the following to configure your port mappings to work with Postgres:
```toml
[[services]]
internal_port = 5432 # Postgres instance
protocol = "tcp"
[[services.ports]]
handlers = ["pg_tls"]
port = 5432
```
Note the use of the `pg_tls` [handler](/docs/networking/services/#connection-handlers) to manage the specific requirements of Postgres connections.
For additional information on services and service ports:
[The services sections](https://fly.io/docs/reference/configuration/#the-services-sections)
## Deploy with the new configuration
Once your service has been set up in `fly.toml`, it's time to deploy with the new configuration.
Verify the version of Postgres you are running. **This step is important, because [there can be changes in the internal storage format between major versions of Postgres](https://www.postgresql.org/docs/current/upgrading.html).**
Figure out which image and tag (Postgres version) you’re on:
```cmd
fly image show --app <pg-app-name>
```
```out
Image Details
Registry = docker-hub-mirror.fly.io
Repository = flyio/postgres-flex
Tag = 16.4
Version = v0.0.62
```
Deploy your cluster, using `--image` with the `image:tag` found in the previous step:
```cmd
fly deploy . --app <pg-app-name> --image flyio/postgres-flex:<major-version>
```
As an example, if you are running Postgres 16.x you would specify `flyio/postgres-flex:16` as your target image.
After the deployment completes, you can verify your `services` configuration by running the `fly services list` command:
```cmd
fly services list
```
```output
Services
PROTOCOL PORTS FORCE HTTPS
TCP 5432 => 5432 [PG_TLS] False
TCP 5433 => 5433 [PG_TLS] False
```
You should then be able to access your Postgres cluster via psql like:
```cmd
psql "sslmode=require host=<pg-app-name>.fly.dev dbname=<db name> user=<username>"
```
```out
Password for user <username>:
psql (14.5 (Homebrew), server 13.6 (Debian 13.6-1.pgdg110+1))
SSL connection (protocol: TLSv1.3, cipher: TLS_AES_256_GCM_SHA384, bits: 256, compression: off)
Type "help" for help.
<db name>=#
```
## Adapting the connection string
The connection string that `fly pg create` outputs for use in consuming apps is in the form:
```
postgres://{username}:{password}@{hostname}:{port}/{database}?options
```
where the `hostname` is an internal one. Substitute your newly publicly reachable hostname (`<pg-app-name>.fly.dev`) here to get a connection string an external app can use.
External Connections
Fly Postgres databases can be used by applications outside their Fly.io internal private network; this means in a different private network belonging to your organization, in another Fly organization, or outside Fly.io altogether.
We don’t expose Postgres apps to the internet by default. To get this working, you’ll need to make two adaptations: configuring your Postgres app to accept connections from the Fly proxy, and providing a publicly-resolvable hostname to your app.
Allocate an IP address
If you haven’t already, you will need to allocate a public IP address to your Postgres app. You can view your list of IPs by running the following command from your application directory:
fly ips list --app <pg-app-name>
You can allocate an IPv4 address by running the following:
fly ips allocate-v4 --app <pg-app-name>
If your network supports IPv6:
fly ips allocate-v6 --app <pg-app-name>
Configure an external service
Now that you have an IP address, it’s time to configure your app to accept connections on an external port, and direct incoming requests to your Postgres instance.
Pull down a fly.toml configuration file for your Postgres app, if you don’t have it:
fly config save --app <pg-app-name>
Note that this could overwrite a fly.toml in the current directory, so be careful!
Open up your fly.toml file.
This may come with a default services section for internal_port 8080. Replace that with the following to configure your port mappings to work with Postgres:
where the hostname is an internal one. Substitute your newly publicly reachable hostname (<pg-app-name>.fly.dev) here to get a connection string an external app can use.