Python SDK Reference¶
The Interactive Python SDK Reference is a comprehensive guide designed to assist developers in integrating the Interactive service into their Python applications. This SDK allows users to seamlessly connect to Interactive and harness its powerful features for graph management, stored procedure management, and query execution.
Requirements.¶
Python >= 3.8
Installation & Usage¶
Note
If you want to isolate the installation from your local Python environment, you might consider using a virtual environment virtualenv to create a new one.”
pip install¶
pip3 install gs_interactive
Then import the package:
import gs_interactive
Setuptools¶
Install via Setuptools.
python3 setup.py build_proto
python3 setup.py install --user
(or sudo python3 setup.py install to install the package for all users)
Then import the package:
import gs_interactive
Tests¶
Execute pytest to run the tests.
Getting Started¶
First, install and start the interactive service via Interactive Getting Started, and you will get the all the endpoints for the Interactive service.
You can connect to Interactive service with Interactive SDK, with following environment variables declared.
############################################################################################
    export INTERACTIVE_ADMIN_ENDPOINT=http://127.0.0.1:7777
    export INTERACTIVE_STORED_PROC_ENDPOINT=http://127.0.0.1:10000
    export INTERACTIVE_CYPHER_ENDPOINT=neo4j://127.0.0.1:7687
############################################################################################
Note
If you have customized the ports when deploying Interactive, remember to replace the default ports with your customized ports.
Remember to export these environment variables.
Connect and submit a query¶
Before connecting to the Interactive Service and submitting queries via the Python SDK, ensure that the service is running on the intended graph.
gsctl use GRAPH <graph_name>
gsctl service status
Now submit query via Interactive Python SDK.
from gs_interactive.client.driver import Driver
from gs_interactive.client.session import Session
from gs_interactive.models import *
driver = Driver()
with driver.getNeo4jSession() as session:
    result = session.run("MATCH(n) RETURN COUNT(n);")
    for record in result:
        print(record)
Create a new graph¶
To create a new graph, user need to specify the name, description, vertex types and edges types. For the detail data model of the graph, please refer to Data Model.
In this example, we will create a simple graph with only one vertex type persson, and one edge type named knows.
def create_graph(sess : Session):
    # Define the graph schema via a python dict.
    test_graph_def = {
        "name": "test_graph",
        "description": "This is a test graph",
        "schema": {
            "vertex_types": [
                {
                    "type_name": "person",
                    "properties": [
                        {
                            "property_name": "id",
                            "property_type": {"primitive_type": "DT_SIGNED_INT64"},
                        },
                        {
                            "property_name": "name",
                            "property_type": {"string": {"long_text": ""}},
                        },
                        {
                            "property_name": "age",
                            "property_type": {"primitive_type": "DT_SIGNED_INT32"},
                        },
                    ],
                    "primary_keys": ["id"],
                }
            ],
            "edge_types": [
                {
                    "type_name": "knows",
                    "vertex_type_pair_relations": [
                        {
                            "source_vertex": "person",
                            "destination_vertex": "person",
                            "relation": "MANY_TO_MANY",
                        }
                    ],
                    "properties": [
                        {
                            "property_name": "weight",
                            "property_type": {"primitive_type": "DT_DOUBLE"},
                        }
                    ],
                    "primary_keys": [],
                }
            ],
        },
    }
    create_graph_request = CreateGraphRequest.from_dict(test_graph_def)
    resp = sess.create_graph(create_graph_request)
    assert resp.is_ok()
    return resp.get_value().graph_id
driver = Driver()
sess = driver.session()
graph_id = create_graph(sess)
print("Created graph, id is ", graph_id)
In the aforementioned example, a graph named test_graph is defined using a python dictionaly. You can also define the graph using the programmatic interface provided by CreateGraphRequest. Upon calling the createGraph method, a string representing the unique identifier of the graph is returned.
Note
You might observe that we define the graph schema in YAML with gsctl, but switch to using dict in Python code. You may encounter challenges when converting between different formats.
However, converting YAML to a Python dict is quite convenient.
First, install pyYAML
pip3 install pyYAML
Then use pyYAML to convert the YAML string to a Python dict
import yaml
yaml_string = """
...
"""
python_dict = yaml.safe_load(yaml_string)
print(python_dict)
Afterwards, you can create a CreateGraphRequest from the Python dict.
Import data to the graph¶
After a new graph is created, you may want to import data into the newly created graph. For the detail configuration of data import, please refer to Data Import Configuration.
For example, you can import the local csv files into the test_graph. Note that, currently only csv files are supported now. The raw data of test_graph is available at GraphScope Interactive Github repo, and you can download them with the following command.
wget https://raw.githubusercontent.com/alibaba/GraphScope/main/flex/interactive/examples/modern_graph/person.csv
wget https://raw.githubusercontent.com/alibaba/GraphScope/main/flex/interactive/examples/modern_graph/person_knows_person.csv
Now run the following code in python Interpreter, remember to replace /path/to/person.csv and /path/to/person_knows_person.csv with the actual local path.
def bulk_loading(sess: Session, graph_id : str):
    test_graph_datasource = {
        "loading_config":{
            "data_source":{
                "scheme": "file"
            },
            "import_option": "init",
            "format":{
                "type": "csv"
            },
        },
        "vertex_mappings": [
            {
                "type_name": "person",
                "inputs": ["@/path/to/person.csv"],
                "column_mappings": [
                    {"column": {"index": 0, "name": "id"}, "property": "id"},
                    {"column": {"index": 1, "name": "name"}, "property": "name"},
                    {"column": {"index": 2, "name": "age"}, "property": "age"},
                ],
            }
        ],
        "edge_mappings": [
            {
                "type_triplet": {
                    "edge": "knows",
                    "source_vertex": "person",
                    "destination_vertex": "person",
                },
                "inputs": [
                    "@/path/to/person_knows_person.csv"
                ],
                "source_vertex_mappings": [
                    {"column": {"index": 0, "name": "person.id"}, "property": "id"}
                ],
                "destination_vertex_mappings": [
                    {"column": {"index": 1, "name": "person.id"}, "property": "id"}
                ],
                "column_mappings": [
                    {"column": {"index": 2, "name": "weight"}, "property": "weight"}
                ],
            }
        ],
    }
    bulk_load_request = SchemaMapping.from_dict(test_graph_datasource)
    resp = sess.bulk_loading(graph_id, bulk_load_request)
    assert resp.is_ok()
    job_id = resp.get_value().job_id
    print('The bulkloading job id is ', job_id)
    # wait until the job has completed successfully.
    while True:
        resp = sess.get_job(job_id)
        assert resp.is_ok()
        status = resp.get_value().status
        print("job status: ", status)
        if status == "SUCCESS":
            break
        elif status == "FAILED":
            raise Exception("job failed")
        else:
            time.sleep(1)
# We assume you have created the graph, uncomment 
# following code if you haven't create the new graph yet.
# 
# driver = Driver()
# sess = driver.session()
# graph_id = create_graph(sess)
# print("Created graph, id is ", graph_id)
# 
bulk_loading(sess, graph_id)
For each vertex/edge types, you need to provide the input data source and column mapping information.
Remember to add @ at the beginning of the local file path.
Session.bulkLoading() will submit an data loading job to the service, and we can query the status of the job via Session.getJobStatus(), and wait until the job has compleleted successfully.
Create a stored procedure¶
Stored procedures can be registered into GraphScope Interactive to encapsulate and reuse complex graph operations. Interactive support both cypher and c++ queries as stored procedures.
With the following code, you will create a procedure named testProcedure which is defined via a cypher query.
# Create Graph 
# ...
# Bulk loading
# ...
proc_name="test_procedure"
create_proc_request = CreateProcedureRequest(
    name=proc_name,
    description="test procedure",
    query="MATCH (n) RETURN COUNT(n);",
    type="cypher",
)
resp = sess.create_procedure(graph_id, create_proc_request)
assert resp.is_ok()
print("successfully create procedure: ", proc_name)
The procedure could not be invoked now, since currently interactive service has not been switched to the newly created modern_graph. We need to start the service on modern_graph.
Start the query service on the new graph¶
Although Interactive supports multiple graphs in terms of logic and storage, it can only serve on one graph at a time. This means that at any given moment, only one graph is available to provide query service. So we need to switch to the newly created modern_graph with following code.
resp = sess.start_service(
    start_service_request=StartServiceRequest(graph_id=graph_id)
)
assert resp.is_ok()
print("start service ok", resp)
Submit queries to the new graph¶
After starting query service on the new graph, we are now able to submit queries to modern_graph.
Submit cypher queries¶
query = "MATCH (n) RETURN COUNT(n);"
with driver.getNeo4jSession() as session:
    resp = session.run(query)
    for record in resp:
        print(record)
Delete the graph¶
Finally, we can delete the graph. The graph data and stored procedure bound to the graph will also be deleted. Currently Interactive forbid deleting a graph which is currently serving in the service, so to delete graph, stop the service first.
# stop the service first, note that graph_id is optional.
resp = sess.stop_service(graph_id)
assert resp.is_ok()
print("successfully stopped the service")
resp = sess.delete_graph(graph_id)
assert resp.is_ok()
print("delete graph res: ", resp)
For the full example, please refer to Python SDK Example
Documentation for Service APIs¶
The Service APIs in interactive SDK are divided into five categories.
- GraphManagementApi 
- ProcedureManagementApi 
- JobManagementApi 
- ServiceManagementApi 
- QueryServiceApi 
- VertexApi 
- EdgeApi 
All URIs are relative to ${INTERACTIVE_ADMIN_ENDPOINT}
| Class | Method | HTTP request | Description | 
|---|---|---|---|
| GraphManagementApi | POST /v1/graph/{graph_id}/dataloading | ||
| GraphManagementApi | POST /v1/graph | ||
| GraphManagementApi | DELETE /v1/graph/{graph_id} | ||
| GraphManagementApi | GET /v1/graph/{graph_id} | ||
| GraphManagementApi | GET /v1/graph/{graph_id}/schema | ||
| GraphManagementApi | GET /v1/graph | ||
| GraphManagementApi | GET /v1/graph/{graph_id}/statistics | ||
| JobManagementApi | DELETE /v1/job/{job_id} | ||
| JobManagementApi | GET /v1/job/{job_id} | ||
| JobManagementApi | GET /v1/job | ||
| ProcedureManagementApi | POST /v1/graph/{graph_id}/procedure | ||
| ProcedureManagementApi | DELETE /v1/graph/{graph_id}/procedure/{procedure_id} | ||
| ProcedureManagementApi | GET /v1/graph/{graph_id}/procedure/{procedure_id} | ||
| ProcedureManagementApi | GET /v1/graph/{graph_id}/procedure | ||
| ProcedureManagementApi | PUT /v1/graph/{graph_id}/procedure/{procedure_id} | ||
| ServiceManagementApi | GET /v1/service/status | ||
| ServiceManagementApi | POST /v1/service/restart | ||
| ServiceManagementApi | POST /v1/service/start | ||
| ServiceManagementApi | POST /v1/service/stop | ||
| QueryServiceApi | POST /v1/graph/{graph_id}/query | ||
| QueryServiceApi | POST /v1/graph/current/query | ||
| VertexApi | POST /v1/graph/{graph_id}/vertex | Add vertex to the graph | |
| VertexApi | GET /v1/graph/{graph_id}/vertex | Get the vertex’s properties with vertex primary key. | |
| VertexApi | PUT /v1/graph/{graph_id}/vertex | Update vertex’s property | |
| EdgeApi | POST /v1/graph/{graph_id}/edge | Add edge to the graph | |
| EdgeApi | GET /v1/graph/{graph_id}/edge | Get the edge’s properties with src and dst vertex primary keys. | |
| EdgeApi | PUT /v1/graph/{graph_id}/edge | Update edge’s property |