Recently, I embarked on a project to deploy Neo4j on Azure Kubernetes Service (AKS), aiming to leverage its powerful graph database capabilities within a cloud-native environment. Everything was running smoothly until I encountered a challenge: external application(which sit inside another cluster) needed to access Neo4j via the Bolt protocol, which operates on port 7687.
In this post, I will walk you through the process of setting up an Ingress for Neo4j in a Kubernetes environment, specifically focusing on exposing the 7687 Bolt port. This blog assumes that Neo4j is deployed using the official Neo4j Helm chart and that Nginx is installed using the Nginx Ingress Controller Helm chart.
Solution
The manual steps are outlined in the Neo4j documentation, specifically under Access you data via Cypher Shell . However, since we used the Nginx Ingress Controller Helm chart to deploy the ingress controller, it automated all the steps defined in the documentation by supplying it with helm configuration parameters.
-
Identify the Neo4j service that exposes the Bolt service and its namespace. In my case, the service that exposes the Bolt port (7687) is
neo4j-cluster-lb-neo4j
, which resides in theneo4j
namespace. -
Apply the following helm configuration value on Nginx Ingress Controller.
1 2 3 4
nginx: enabled: true tcp: 7687: "neo4j/neo4j-cluster-lb-neo4j:7687"
Validation
-
Then, we can validate the port 7687 is exposed by using telnet.
-
After validating that the port is exposed through the ingress, we can proceed to verify that the application can actually access Neo4j. Below is a sample application to test the connection.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
public static void main(String[] args) { // Update the URI to include the correct scheme if needed String uri = "neo4j+ssc://<ingress url>/"; String user = "<user>"; String password = "<password>"; // Create the driver instance try (Driver driver = GraphDatabase.driver(uri, AuthTokens.basic(user, password));Session session = driver.session()) { driver.verifyConnectivity(); session.executeRead(tx -> { List<Record> records = tx.run("show databases").list(); for (Record record : records) { System.out.println("name :" + record.get("name")); System.out.println("address :" + record.get("address") ); System.out.println("requestedStatus :" + record.get("requestedStatus")); System.out.println("-----------"); } return null; }); } catch (Exception e) { e.printStackTrace(); System.out.println("Error connecting to Neo4j: " + e.getMessage()); } }
Below is a sample response from the program if everything is configured correctly:
1 2 3 4 5 6 7 8 9 10 11 12
name :"system" address :"server-2.neo4j.svc.cluster.local:7687" requestedStatus :"online" ----------- name :"system" address :"server-1.neo4j.svc.cluster.local:7687" requestedStatus :"online" ----------- name :"system" address :"server-4.neo4j.svc.cluster.local:7687" requestedStatus :"online" -----------