diff --git a/Taskfile.yml b/Taskfile.yml index 839bd859dfdd..08a4159fcffc 100644 --- a/Taskfile.yml +++ b/Taskfile.yml @@ -17,7 +17,6 @@ vars: FUZZ_CORPUS: ../fuzz-corpus TESTJS_PORT: 27017 RACE_FLAG: -race={{and (ne OS "windows") (ne ARCH "arm") (ne ARCH "riscv64")}} - TEST_UNIX_SOCKET_FLAG: -target-unix-socket={{ne OS "windows"}} BUILD_TAGS: ferretdb_debug,ferretdb_hana SERVICES: postgres postgres_secured mongodb mongodb_secured jaeger @@ -252,9 +251,12 @@ tasks: go test -count=1 -run='{{or .TEST_RUN .SHARD_RUN}}' -timeout={{.TEST_TIMEOUT}} {{.RACE_FLAG}} -tags={{.BUILD_TAGS}} -shuffle=on -coverpkg=../... -coverprofile=integration-hana.txt . -target-backend=ferretdb-hana - {{.TEST_UNIX_SOCKET_FLAG}} + -target-tls -hana-url=$FERRETDB_HANA_URL - -compat-url='mongodb://127.0.0.1:47017/' + -compat-url='mongodb://username:password@127.0.0.1:47018/?tls=true&tlsCertificateKeyFile=../build/certs/client.pem&tlsCaFile=../build/certs/rootCA-cert.pem' + -use-new-hana + -disable-filter-pushdown={{.DISABLE_FILTER_PUSHDOWN}} + -enable-sort-pushdown={{.ENABLE_SORT_PUSHDOWN}} vars: SHARD_RUN: sh: go run -C .. ./cmd/envtool tests shard --index={{.SHARD_INDEX}} --total={{.SHARD_TOTAL}} @@ -315,6 +317,7 @@ tasks: desc: "Run FerretDB with `pg` handler" deps: [build-host] cmds: + # TODO --use-new-pg - > bin/ferretdb{{exeExt}} --listen-addr=:27017 @@ -344,6 +347,7 @@ tasks: desc: "Run FerretDB with `hana` handler" deps: [build-host] cmds: + # TODO --use-new-hana - > bin/ferretdb{{exeExt}} --listen-addr=:27017 @@ -356,6 +360,7 @@ tasks: desc: "Run FerretDB with `pg` handler (TLS, auth required)" deps: [build-host] cmds: + # TODO --use-new-pg - > bin/ferretdb{{exeExt}} --listen-addr='' @@ -373,6 +378,7 @@ tasks: desc: "Run FerretDB in diff-proxy mode" deps: [build-host] cmds: + # TODO --use-new-pg - > bin/ferretdb{{exeExt}} --listen-addr=:27017 diff --git a/cmd/ferretdb/main.go b/cmd/ferretdb/main.go index f3f5b5f4efce..21e04f581acd 100644 --- a/cmd/ferretdb/main.go +++ b/cmd/ferretdb/main.go @@ -84,7 +84,8 @@ var cli struct { EnableSortPushdown bool `default:"false" help:"Experimental: enable sort pushdown."` EnableOplog bool `default:"false" help:"Experimental: enable OpLog." hidden:""` - UseNewPG bool `default:"false" help:"Experimental: use new PostgreSQL backend." hidden:""` + UseNewPG bool `default:"false" help:"Experimental: use new PostgreSQL backend." hidden:""` + UseNewHana bool `default:"false" help:"Experimental: use new SAP HANA backend." hidden:""` //nolint:lll // for readability Telemetry struct { @@ -367,7 +368,8 @@ func run() { EnableSortPushdown: cli.Test.EnableSortPushdown, EnableOplog: cli.Test.EnableOplog, - UseNewPG: cli.Test.UseNewPG, + UseNewPG: cli.Test.UseNewPG, + UseNewHana: cli.Test.UseNewHana, }, }) if err != nil { diff --git a/integration/setup/listener.go b/integration/setup/listener.go index 13e346ba43d8..1a7666320d5d 100644 --- a/integration/setup/listener.go +++ b/integration/setup/listener.go @@ -178,7 +178,8 @@ func setupListener(tb testtb.TB, ctx context.Context, logger *zap.Logger) string EnableSortPushdown: *enableSortPushdownF, EnableOplog: *enableOplogF, - UseNewPG: *useNewPGF, + UseNewPG: *useNewPgF, + UseNewHana: *useNewHanaF, }, } h, err := registry.NewHandler(handler, handlerOpts) diff --git a/integration/setup/setup.go b/integration/setup/setup.go index 19d014544bac..9b995a1188bc 100644 --- a/integration/setup/setup.go +++ b/integration/setup/setup.go @@ -63,7 +63,8 @@ var ( enableSortPushdownF = flag.Bool("enable-sort-pushdown", false, "enable sort pushdown") enableOplogF = flag.Bool("enable-oplog", false, "enable OpLog") - useNewPGF = flag.Bool("use-new-pg", false, "use new PostgreSQL backend") + useNewPgF = flag.Bool("use-new-pg", false, "use new PostgreSQL backend") + useNewHanaF = flag.Bool("use-new-hana", false, "use new SAP HANA backend") ) // Other globals. diff --git a/internal/backends/hana/backend.go b/internal/backends/hana/backend.go new file mode 100644 index 000000000000..60642b205774 --- /dev/null +++ b/internal/backends/hana/backend.go @@ -0,0 +1,90 @@ +// Copyright 2021 FerretDB Inc. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package hana + +import ( + "context" + + "github.com/prometheus/client_golang/prometheus" + "go.uber.org/zap" + + "github.com/FerretDB/FerretDB/internal/backends" + "github.com/FerretDB/FerretDB/internal/util/lazyerrors" + "github.com/FerretDB/FerretDB/internal/util/state" +) + +// backend implements backends.Backend interface. +type backend struct{} + +// NewBackendParams represents the parameters of NewBackend function. +// +//nolint:vet // for readability +type NewBackendParams struct { + URI string + L *zap.Logger + P *state.Provider +} + +// NewBackend creates a new backend. +func NewBackend(params *NewBackendParams) (backends.Backend, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// Close implements backends.Backend interface. +func (b *backend) Close() { + // TODO +} + +// Name implements backends.Backend interface. +func (b *backend) Name() string { + return "SAP HANA" +} + +// Status implements backends.Backend interface. +func (b *backend) Status(ctx context.Context, params *backends.StatusParams) (*backends.StatusResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// Database implements backends.Backend interface. +func (b *backend) Database(name string) (backends.Database, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// ListDatabases implements backends.Backend interface. +// +//nolint:lll // for readability +func (b *backend) ListDatabases(ctx context.Context, params *backends.ListDatabasesParams) (*backends.ListDatabasesResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// DropDatabase implements backends.Backend interface. +func (b *backend) DropDatabase(ctx context.Context, params *backends.DropDatabaseParams) error { + return lazyerrors.New("not implemented yet") +} + +// Describe implements prometheus.Collector. +func (b *backend) Describe(ch chan<- *prometheus.Desc) { + // TODO +} + +// Collect implements prometheus.Collector. +func (b *backend) Collect(ch chan<- prometheus.Metric) { + // TODO +} + +// check interfaces +var ( + _ backends.Backend = (*backend)(nil) +) diff --git a/internal/backends/hana/collection.go b/internal/backends/hana/collection.go new file mode 100644 index 000000000000..53fa7afb0727 --- /dev/null +++ b/internal/backends/hana/collection.go @@ -0,0 +1,75 @@ +// Copyright 2021 FerretDB Inc. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package hana + +import ( + "context" + + "github.com/FerretDB/FerretDB/internal/backends" + "github.com/FerretDB/FerretDB/internal/util/lazyerrors" +) + +// collection implements backends.Collection interface by delegating all methods to the wrapped database. +type collection struct{} + +// Query implements backends.Collection interface. +func (c *collection) Query(ctx context.Context, params *backends.QueryParams) (*backends.QueryResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// InsertAll implements backends.Collection interface. +func (c *collection) InsertAll(ctx context.Context, params *backends.InsertAllParams) (*backends.InsertAllResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// UpdateAll implements backends.Collection interface. +func (c *collection) UpdateAll(ctx context.Context, params *backends.UpdateAllParams) (*backends.UpdateAllResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// DeleteAll implements backends.Collection interface. +func (c *collection) DeleteAll(ctx context.Context, params *backends.DeleteAllParams) (*backends.DeleteAllResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// Explain implements backends.Collection interface. +func (c *collection) Explain(ctx context.Context, params *backends.ExplainParams) (*backends.ExplainResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// Stats implements backends.Collection interface. +func (c *collection) Stats(ctx context.Context, params *backends.CollectionStatsParams) (*backends.CollectionStatsResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// ListIndexes implements backends.Collection interface. +func (c *collection) ListIndexes(ctx context.Context, params *backends.ListIndexesParams) (*backends.ListIndexesResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// CreateIndexes implements backends.Collection interface. +func (c *collection) CreateIndexes(ctx context.Context, params *backends.CreateIndexesParams) (*backends.CreateIndexesResult, error) { //nolint:lll // for readability + return nil, lazyerrors.New("not implemented yet") +} + +// DropIndexes implements backends.Collection interface. +func (c *collection) DropIndexes(ctx context.Context, params *backends.DropIndexesParams) (*backends.DropIndexesResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// check interfaces +var ( + _ backends.Collection = (*collection)(nil) +) diff --git a/internal/backends/hana/database.go b/internal/backends/hana/database.go new file mode 100644 index 000000000000..9dbf858c3253 --- /dev/null +++ b/internal/backends/hana/database.go @@ -0,0 +1,62 @@ +// Copyright 2021 FerretDB Inc. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package hana + +import ( + "context" + + "github.com/FerretDB/FerretDB/internal/backends" + "github.com/FerretDB/FerretDB/internal/util/lazyerrors" +) + +// database implements backends.Database. +type database struct{} + +// Collection implements backends.Database interface. +func (db *database) Collection(name string) (backends.Collection, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// ListCollections implements backends.Database interface. +// +//nolint:lll // for readability +func (db *database) ListCollections(ctx context.Context, params *backends.ListCollectionsParams) (*backends.ListCollectionsResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// CreateCollection implements backends.Database interface. +func (db *database) CreateCollection(ctx context.Context, params *backends.CreateCollectionParams) error { + return lazyerrors.New("not implemented yet") +} + +// DropCollection implements backends.Database interface. +func (db *database) DropCollection(ctx context.Context, params *backends.DropCollectionParams) error { + return lazyerrors.New("not implemented yet") +} + +// RenameCollection implements backends.Database interface. +func (db *database) RenameCollection(ctx context.Context, params *backends.RenameCollectionParams) error { + return lazyerrors.New("not implemented yet") +} + +// Stats implements backends.Database interface. +func (db *database) Stats(ctx context.Context, params *backends.DatabaseStatsParams) (*backends.DatabaseStatsResult, error) { + return nil, lazyerrors.New("not implemented yet") +} + +// check interfaces +var ( + _ backends.Database = (*database)(nil) +) diff --git a/internal/backends/hana/dummy_test.go b/internal/backends/hana/dummy_test.go new file mode 100644 index 000000000000..baf26e9328d2 --- /dev/null +++ b/internal/backends/hana/dummy_test.go @@ -0,0 +1,21 @@ +// Copyright 2021 FerretDB Inc. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package hana + +import "testing" + +func TestDummy(t *testing.T) { + // we need at least one test per package to correctly calculate coverage +} diff --git a/internal/backends/hana/hana.go b/internal/backends/hana/hana.go new file mode 100644 index 000000000000..9980d9a37020 --- /dev/null +++ b/internal/backends/hana/hana.go @@ -0,0 +1,16 @@ +// Copyright 2021 FerretDB Inc. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// Package hana provides backend for SAP HANA. +package hana diff --git a/internal/backends/postgresql/backend.go b/internal/backends/postgresql/backend.go index 0fd48d1469dd..c6f4cad1a8ae 100644 --- a/internal/backends/postgresql/backend.go +++ b/internal/backends/postgresql/backend.go @@ -40,12 +40,8 @@ type NewBackendParams struct { P *state.Provider } -// NewBackend creates a new backend for PostgreSQL-compatible database. +// NewBackend creates a new backend. func NewBackend(params *NewBackendParams) (backends.Backend, error) { - if params.P == nil { - panic("state provider is required but not set") - } - r, err := metadata.NewRegistry(params.URI, params.L, params.P) if err != nil { return nil, err diff --git a/internal/backends/postgresql/dummy_test.go b/internal/backends/postgresql/dummy_test.go new file mode 100644 index 000000000000..ab1be0278a87 --- /dev/null +++ b/internal/backends/postgresql/dummy_test.go @@ -0,0 +1,21 @@ +// Copyright 2021 FerretDB Inc. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package postgresql + +import "testing" + +func TestDummy(t *testing.T) { + // we need at least one test per package to correctly calculate coverage +} diff --git a/internal/backends/postgresql/doc.go b/internal/backends/postgresql/postgresql.go similarity index 100% rename from internal/backends/postgresql/doc.go rename to internal/backends/postgresql/postgresql.go diff --git a/internal/backends/sqlite/backend.go b/internal/backends/sqlite/backend.go index 65cfe992bbf3..a34d54e8e9fd 100644 --- a/internal/backends/sqlite/backend.go +++ b/internal/backends/sqlite/backend.go @@ -40,12 +40,8 @@ type NewBackendParams struct { P *state.Provider } -// NewBackend creates a new SQLite backend. +// NewBackend creates a new backend. func NewBackend(params *NewBackendParams) (backends.Backend, error) { - if params.P == nil { - panic("state provider is required but not set") - } - r, err := metadata.NewRegistry(params.URI, params.L, params.P) if err != nil { return nil, err diff --git a/internal/handlers/registry/hana.go b/internal/handlers/registry/hana.go index 56ee05f7fffb..a3209c7f63d6 100644 --- a/internal/handlers/registry/hana.go +++ b/internal/handlers/registry/hana.go @@ -19,6 +19,7 @@ package registry import ( "github.com/FerretDB/FerretDB/internal/handlers" "github.com/FerretDB/FerretDB/internal/handlers/hana" + "github.com/FerretDB/FerretDB/internal/handlers/sqlite" ) // init registers "hana" handler for Hana when "ferretdb_hana" build tag is provided. @@ -26,6 +27,23 @@ func init() { registry["hana"] = func(opts *NewHandlerOpts) (handlers.Interface, error) { opts.Logger.Warn("HANA handler is in alpha. It is not supported yet.") + if opts.UseNewHana { + handlerOpts := &sqlite.NewOpts{ + Backend: "hana", + URI: opts.HANAURL, + + L: opts.Logger.Named("hana"), + ConnMetrics: opts.ConnMetrics, + StateProvider: opts.StateProvider, + + DisableFilterPushdown: opts.DisableFilterPushdown, + EnableSortPushdown: opts.EnableSortPushdown, + EnableOplog: opts.EnableOplog, + } + + return sqlite.New(handlerOpts) + } + handlerOpts := &hana.NewOpts{ HANAURL: opts.HANAURL, L: opts.Logger, diff --git a/internal/handlers/registry/pg.go b/internal/handlers/registry/pg.go index a063b77efb2f..eab411c03b94 100644 --- a/internal/handlers/registry/pg.go +++ b/internal/handlers/registry/pg.go @@ -24,8 +24,6 @@ import ( func init() { registry["pg"] = func(opts *NewHandlerOpts) (handlers.Interface, error) { if opts.UseNewPG { - opts.Logger.Warn("New PostgreSQL backend is in alpha.") - handlerOpts := &sqlite.NewOpts{ Backend: "postgresql", URI: opts.PostgreSQLURL, diff --git a/internal/handlers/registry/registry.go b/internal/handlers/registry/registry.go index b555190dd6d5..7c0babed6531 100644 --- a/internal/handlers/registry/registry.go +++ b/internal/handlers/registry/registry.go @@ -59,7 +59,8 @@ type TestOpts struct { EnableSortPushdown bool EnableOplog bool - UseNewPG bool + UseNewPG bool + UseNewHana bool } // NewHandler constructs a new handler. diff --git a/internal/handlers/sqlite/sqlite.go b/internal/handlers/sqlite/sqlite.go index 74ec26faf905..7323b4473c36 100644 --- a/internal/handlers/sqlite/sqlite.go +++ b/internal/handlers/sqlite/sqlite.go @@ -23,6 +23,7 @@ import ( "github.com/FerretDB/FerretDB/internal/backends" "github.com/FerretDB/FerretDB/internal/backends/decorators/oplog" + "github.com/FerretDB/FerretDB/internal/backends/hana" "github.com/FerretDB/FerretDB/internal/backends/postgresql" "github.com/FerretDB/FerretDB/internal/backends/sqlite" "github.com/FerretDB/FerretDB/internal/clientconn/connmetrics" @@ -75,6 +76,12 @@ func New(opts *NewOpts) (handlers.Interface, error) { L: opts.L, P: opts.StateProvider, }) + case "hana": + b, err = hana.NewBackend(&hana.NewBackendParams{ + URI: opts.URI, + L: opts.L, + P: opts.StateProvider, + }) default: panic("unknown backend: " + opts.Backend) }