/structor

The structor project is component of QuanXiang, it's a abstract layer between bussiness layer and database.

Primary LanguageGoApache License 2.0Apache-2.0

Structor

Overview

The structor project is component of QuanXiang, it's a abstract layer between bussiness layer and database, that will make users easy to use database without database knowledge. We use json to define table's columns data type

Getting Started

Structor implements GRPC API for services call. gRPC is useful for low-latency, high performance scenarios and has language integration using the proto clients. If you want to use Structor for your project, refer GRPC for more information about GRPC. Structor can be started up as a service or a sidecar with Kubernetes pod.

Installation

git clone https://github.com/quanxiang-cloud/structor.git
cd structor
CGO_ENABLED=1 GOARCH=amd64 GOOS=Linux go build -tags mysql -o structor cmd/structor/main.go

Notice:

The Build example base on Linux-like OS, if you use other OS, please change the GOOS and GOARCH for your case.

  • GOOS: darwin, linux, windows, freebsd etc.
  • GOARCH: amd64, arm, i386, p360 etc.

How to run

For MySQL connection, command example as below:

./structor --mysql-host=127.0.0.1:3306 --mysql-database=example --mysql-user=root --mysql-password=***** --mysql-log=false --mysql-log-level=0 --mysql-maxIdelConns=10 --mysql-maxOpenConns=20 --mysql-engine=innoDB --my-charset=utf8 --mysql-collate=utf8_unicode_ci

For MongoDB connection, command example as below:

./structor --mongo-host=127.0.0.1:27017 --mongo-direct=true --mongo-auth-mechanism=SCRAM-SHA-1 --mongo-auth-source=admin --mongo-password=***** --mongo-password-set=false --mongo-database=example

How to use

Structor supports Data Definition Statements with DDL and Data Manipulation Statements with DSL. We refer ElasticSearch Query DSL to define our DSL, we just support a little of ElasticSearch queries.

Below code will be use in all code example, just replace "// Add code here" in your case.

package main

import (
  "context"
	"encoding/json"
	"fmt"
	
	client "github.com/quanxiang-cloud/structor/api/proto"

	"google.golang.org/grpc"
	"google.golang.org/protobuf/types/known/anypb"
	"google.golang.org/protobuf/types/known/structpb"
)

func main() {
  dsl, ddl, err := getConn("localhost:8080")
	if err != nil {
		panic(err)
	}
	// Add code here
	_, _ = dsl, ddl
}

func getConn(addr string) (client.DSLServiceClient, client.DDLServiceClient, error) {
	conn, err := grpc.Dial(addr, grpc.WithInsecure())
	if err != nil {
		return nil, nil, err
	}

	dslConn := client.NewDSLServiceClient(conn)
	ddlConn := client.NewDDLServiceClient(conn)
	return dslConn, ddlConn, nil
}

func anyToRaw(any *anypb.Any) (json.RawMessage, error) {
	out := structpb.NewNullValue()
	err := any.UnmarshalTo(out)
	if err != nil {
		return nil, err
	}

	body, err := out.MarshalJSON()
	if err != nil {
		return nil, err
	}
	return body, nil
}

func rawToAny(raw []byte) (*anypb.Any, error) {
	in := structpb.NewNullValue()
	err := in.UnmarshalJSON(raw)
	if err != nil {
		return nil, err
	}
	any := &anypb.Any{}
	err = any.MarshalFrom(in)
	return any, err
}
Data Definition Statements

Create Table:

  _, err := ddl.Create(context.Background(), &client.CreateReq{
		TableName: "idtest",
		Fields: []*client.Field{
			{
				Title:   "bool",
				Type:    "bool",
				Comment: "bool",
			},
		},
	})
	if err != nil {
		panic(err)
	}

Add column definition:

  _, err := ddl.Add(context.Background(), &client.AddReq{
		TableName: "idtest",
		Fields: []*client.Field{
			{
				Title: "object",
				Type:  "object",
			},
			{
				Title: "date",
				Type:  "datetime",
			},
			{
				Title: "float",
				Max:   321321,
				Type:  "float",
			},
			{
				Title: "varchar",
				Max:   200,
				Type:  "string",
			},
			{
				Title: "text",
				Max:   500,
				Type:  "string",
			},
		},
	})
	if err != nil {
		panic(err)
	}

Update table:

	_, err := ddl.Modify(context.Background(), &client.ModifyReq{
		TableName: "idtest",
		Fields: []*client.Field{
			{
				Title: "float",
				Max:   321321,
				Type:  "int",
			},
			{
				Title: "varchar",
				Max:   200,
				Type:  "int",
			},
			{
				Title: "text",
				Max:   500,
				Type:  "int",
			},
		},
	})
	if err != nil {
		panic(err)
	}
Data Manipulation Statements

Query match records :

  queryBody, err := rawToAny([]byte(`{
		"query": {
			"match": {
				"name": "st"
			}
		}
	}`))
	if err != nil {
		panic(err)
	}

	resp, err := dsl.FindOne(context.Background(), &client.FindOneReq{
		TableName: "user",
		Dsl:       queryBody,
	})
	if err != nil {
		panic(err)
	}

	body, err := anyToRaw(resp.GetData())
	if err != nil {
		panic(err)
	}
	fmt.Println(string(body))

Query records:

  any, err := rawToAny([]byte(`
	{
		"query": {
			"bool": {
				"should": [
					{
						"term": {
							"id": "129"
						}
					},
					{
						"bool": {
							"should": [
								{
									"term": {
										"name": "test2"
									}
								}
							]
						}
					}
				]
			}
		}
	}
	`))
	if err != nil {
		panic(err)
	}

	resp, err := dsl.Find(context.Background(), &client.FindReq{
		TableName: "user",
		Dsl:       any,
		Sort:      []string{"id", "-name"},
	})
	if err != nil {
		panic(err)
	}

	data, err := anyToRaw(resp.GetData())
	if err != nil {
		panic(err)
	}
	fmt.Println(string(data))

Delete records:

	delBody, err := rawToAny([]byte(`{
		"query": {
			"range": {
				"id": {
					"gt": 131,
					"lt": 135
				}
			}	
		}
	}`))
	if err != nil {
		panic(err)
	}

	resp, err := dsl.Delete(context.Background(), &client.DeleteReq{
		TableName: "user",
		Dsl:       delBody,
	})
	if err != nil {
		panic(err)
	}

	fmt.Println(resp.Count)

Count records:

	cuBody, err := rawToAny([]byte(`{
		"query": {
			"match": {
				"name": "test"
			}
		}
	}`))
	if err != nil {
		panic(err)
	}

	resp, err := dsl.Count(context.Background(), &client.CountReq{
		TableName: "user",
		Dsl:       cuBody,
	})
	if err != nil {
		panic(err)
	}

	fmt.Println(resp.Data)

Insert record:

	entity, err := rawToAny([]byte(`{
		"id": "136"
	}`))
	if err != nil {
		panic(err)
	}

	entity2, err := rawToAny([]byte(`{
		"id": "137"
	}`))
	if err != nil {
		panic(err)
	}

	var entities []*anypb.Any
	entities = append(entities, entity, entity2)
	_, err = dsl.Insert(context.Background(), &client.InsertReq{
		TableName: "new_user",
		Entities:  entities,
	})
	if err != nil {
		panic(err)
	}

Update records

	updateBody, err := rawToAny([]byte(`{
		"query": {
			"term": {
				"name": "test"
			}
		}
	}`))
	if err != nil {
		panic(err)
	}

	entity, err := rawToAny([]byte(`{
		"name": "test909090"
	}`))
	if err != nil {
		panic(err)
	}

	resp, err := dsl.Update(context.Background(), &client.UpdateReq{
		TableName: "user",
		Dsl:       updateBody,
		Entity:    entity,
	})
	if err != nil {
		panic(err)
	}

	fmt.Println(resp.Count)

Roadmap

Folowing DataBases are supported.

Database CRUD index primary key foreign key version
MySQL 5.7 or later
MongoDB 4.0 or later

Plan to support:

Database CRUD index primary key foreign key version
Oracle
PostgreSQL
TiDB
More

Contributing to Structor

Welcome!!!!