Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/disneystreaming/pg2k4j
Postgresql To Kinesis For Java
https://github.com/disneystreaming/pg2k4j
cdc java kinesis kinesis-stream logical-replication postgresql replication
Last synced: 3 days ago
JSON representation
Postgresql To Kinesis For Java
- Host: GitHub
- URL: https://github.com/disneystreaming/pg2k4j
- Owner: disneystreaming
- License: other
- Created: 2018-10-05T15:40:42.000Z (over 6 years ago)
- Default Branch: master
- Last Pushed: 2024-12-14T07:27:44.000Z (about 1 month ago)
- Last Synced: 2025-01-14T10:13:13.834Z (11 days ago)
- Topics: cdc, java, kinesis, kinesis-stream, logical-replication, postgresql, replication
- Language: Java
- Homepage:
- Size: 921 KB
- Stars: 80
- Watchers: 5
- Forks: 25
- Open Issues: 163
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- postgresql-is-awesome - Postgresql To Kinesis For Java - Disney Streaming
README
[![Build Status](https://travis-ci.com/disneystreaming/pg2k4j.svg?branch=master)](https://travis-ci.com/disneystreaming/pg2k4j) [![codecov](https://codecov.io/gh/disneystreaming/pg2k4j/branch/master/graph/badge.svg)](https://codecov.io/gh/disneystreaming/pg2k4j)
# This project is no longer actively maintained by Disney Streaming Services
You are welcome to use and fork this project, and we may occasionally review and merge pull requests submitted from contributors or dependabot/Snyk. However we no longer use this internally and therefore support will be limited.
## pg2k4j
### Postgresql To Kinesis For Java
A tool for publishing inserts, updates, and deletes made on a [Postgresql](https://www.postgresql.org/) database to an [Amazon Kinesis](https://aws.amazon.com/kinesis/) Stream.
pg2k4j may be run as a stand-alone application from the command line, or used as a Java library where its functionality
can be extended and customized.### Getting Started
First, setup your Postgres database to support [logical replication](https://www.postgresql.org/docs/10/static/logical-replication.html) and create an AWS Kinesis Stream.
#### Run pg2k4j as a Stand-alone Application
Download [Docker](https://www.docker.com/get-started) and login with your docker credentials to gain access to the [pg2k4j docker repository](https://hub.docker.com/r/disneystreaming/pg2k4j/).
Then run the command below.
```
docker run -v /path/to/.aws/creds/:/aws_creds
disneystreaming/pg2k4j
--awsconfiglocation=/aws_creds --awsprofile=default
--pgdatabase= --pghost= --pguser= --pgpassword=
--streamname=
```When you observe the below log, pg2k4j is set to publish changes to Kinesis.
```
[main] INFO com.disneystreaming.pg2k4j.SlotReaderKinesisWriter - Consuming from slot pg2k4j
```
#### Use as a Java Librarypg2k4j artifacts are published to [maven central](https://mvnrepository.com/artifact/com.disneystreaming.pg2k4j/pg2k4j)
##### Maven
```
com.disneystreaming.pg2k4j
pg2k4j
LATEST```
##### Gradle
```
compile group: 'com.disneystreaming.pg2k4j', name: 'pg2k4j', version: 'LATEST'
```##### SBT
```
libraryDependencies += "info.pg2k4j" % "pg2k4j" % "LATEST"
```To initialize and begin publishing database changes to Kinesis, create a [SlotReaderKinesisWriter](src/main/java/com/disneystreaming/pg2k4j/SlotReaderKinesisWriter.java)
and call its [runLoop](src/main/java/com/disneystreaming/pg2k4j/SlotReaderKinesisWriter.java#L84) method.### Why We Wrote pg2k4j
pg2k4j is an implementation of a powerful design pattern called [Change Data Capture](https://en.wikipedia.org/wiki/Change_data_capture).
By using pg2k4j, anyone can know the state of your database at any point in time by consuming from the Kinesis Stream.
At DSS we have rapidly changing datasets that many teams need to access and process in their own way. pg2k4j
alleviates the need to grant database access to each team or to stand up an API on top of the dataset. This keeps the load down
on your database, making it possible to max out its write throughput.### Benefits Over Existing Solutions
Before writing pg2k4j, we explored existing solutions. We used [pg2kinesis](https://github.com/handshake/pg2kinesis) but found
that this implementation simply couldn't keep up with the write throughput that we required. As a JVM app, pg2k4j can natively integrate with [Amazon's
Kinesis Producer Library](https://docs.aws.amazon.com/streams/latest/dev/developing-producers-with-kpl.html) allowing it to achieve write speeds of over
1 million records per minute, which is orders of magnitude faster than the performance we observed with its Python
counterpart.### How it Works
##### 1. pg2k4j Opens up a [Logical Replication Slot](https://www.postgresql.org/docs/10/static/logicaldecoding-explanation.html#LOGICALDECODING-REPLICATION-SLOTS) on the Postgresql database.
A replication slot will stream changes made on the database to the listener of the replication slot in the format specified
by the plugin used for that replication slot. By default pg2k4j uses the [wal2json](https://github.com/eulerto/wal2json) plugin
which outputs a json representation of a [SlotMessage](src/main/java/com/disneystreaming/pg2k4j/models/SlotMessage.java) to the
listening thread. Postgres writes all data changes to the [Write Ahead Log](https://www.postgresql.org/docs/10/static/wal-intro.html), which,
as well as ensuring data integrity and crash safety, makes it possible to perform logical replication. Each replication slot maintains a pointer to a position in the WAL, indicating the last sequence number this replication
slot has processed. This pointer allows Postgres to flush all sections of the WAL which occurred before this sequence number. Crucially, if the
application maintaining the replication slot does not update this sequence number, the storage space on the database will fill up because
Postgres won't be able to clear any sections of the WAL. To view this sequence number you can run the below query on your database.```
select * from pg_replication_slots
```Details of how pg2k4j manages this pointer are outlined later in this section.
##### 2. pg2k4j [deserializes](src/main/java/com/disneystreaming/pg2k4j/SlotReaderKinesisWriter.java#L277) the json output sent by the wal2json plugin to a SlotMessage.
This method should be overridden when using any plugin besides wal2json as the contents from the WAL would not be json
representations of a SlotMessage.##### 3. pg2k4j writes this contents to the Kinesis Stream.
First the SlotMessage is turned into a Stream of [UserRecord](https://github.com/awslabs/amazon-kinesis-producer/blob/master/java/amazon-kinesis-producer/src/main/java/com/amazonaws/services/kinesis/producer/UserRecord.java), and then
these UserRecords are written to the stream with a [callback attached](src/main/java/com/disneystreaming/pg2k4j/SlotReaderKinesisWriter.java#L245) that will be invoked once the records make it to the
stream.##### 4. The callback is invoked when the records succeed or fail to make it to the stream.
On a successful write to the stream pg2k4j will [advance the replication slot's sequence number](src/main/java/com/disneystreaming/pg2k4j/SlotReaderCallback.java#L83), indicating
that any data before this point may be flushed by the database. By advancing the sequence number after receiving confirmation
that the record arrived on the stream, pg2k4j guarantees that each data change reaches Kinesis. Even on Postgres restart
or pg2k4j restart this guarantee is preserved.There is one other scenario wherein pg2k4j will advance the sequence number. It's important to note that each Postgres instance
may have many databases, but a replication slot is configured against a single database. In the scenario where
the replication slot database is idle but the other databases are active, it's important that pg2k4j still advances its pointer into
the WAL so that Postgres doesn't hang onto these sections of the WAL. That's why pg2k4j [advances the sequence number after
a certain period of inactivity](src/main/java/com/disneystreaming/pg2k4j/SlotReaderKinesisWriter.java#L204-L206)
which [defaults to 5 minutes](src/main/java/com/disneystreaming/pg2k4j/ReplicationConfiguration.java#L38).### Configuring Infrastructure
This section is a walk through on how to create your Posgresql instance configured for logical replication as an [RDS](https://aws.amazon.com/rds/) instance.
pg2k4j by no means requires that your Postgesql instance is an RDS instance, but since Kinesis is an AWS product, many users
will likely also be running their Postgres instance on AWS. For an example of how to configure a non-RDS instance of Postgres refer
to the integration tests. This section also walks through how to set up a Kinesis Stream, for which pg2k4j requires no sepcial configuration.#### Start AWS RDS Postgres
##### Create a Parameter Group
In AWS console navigate to RDS then parameter groups and create a parameter group for the `postgres10` family.
![Create Parameter Group](exampleImages/parameterGroup.png)
In this parameter group, set the following values:
```bash
rds.logical_replication 1
max_wal_senders 10
max_replication_slots 10
```##### Launch Instance
In AWS console navigate to RDS->Instances and select `Launch Instance`. Follow the creation wizard,
selecting `Postgresql` for the DB engine and `Postgresql 10.3-R1` for the DB engine version.As shown below, associate the parameter group you created in the previous step with this instance.
![Associate Parameter Group](exampleImages/associateParameterGroup.png)
#### Create Kinesis Stream
In AWS console navigate to Kinesis, and create a stream.
![Create Kinesis Stream](exampleImages/setupKinesisStream.png)
### Contributing
Fork the repo and submit a pr with a description detailing what this code does and what bug or feature it addresses. Any methods
containing substantial logic should include javadocs.Be sure that both integration tests and unit pass and that any new code introduced has corresponding [tests](src/test/java/com/disney/pg2k4j). Run unit tests with
```bash
>> mvn clean test
Tests run: 13, Failures: 0, Errors: 0, Skipped: 0
```and integration tests with
```bash
mvn clean verify
Tests run: 2, Failures: 0, Errors: 0, Skipped: 0
```Contributors are required to fill out a CLA in order for us to be allowed to accept contributions. See [CLA-Individual](CLA-Individual.md) or [CLA-Corporate](CLA-Corporate.md) for details.
### Releasing
Releasing is automatic and is driven by tags. To release simply tag the master branch with a Semantic version, e.g. `1.0.0`.This will update the pom with the version, publish to maven, and build and push the Docker image to Dockerhub.