1
0
mirror of https://github.com/mainflux/mainflux.git synced 2025-04-28 13:48:49 +08:00
Mirko Teodorovic 530f925c4d
MF-1346 - Create Groups API - add grouping of entities (#1334)
* remove owner id

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* add users endpoint for retrieving users from group

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove  groups from things and users

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* move groups into auth

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* separate endpoints for users and things

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix problems with retrieving members

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* add groups test

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove groups from users

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove groups from things

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* rename constant

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* add new errors

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove unnecessary constants

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix validation

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* create groups db mock

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* adding tests

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* revert changes to docker related files

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove groups endpoints from users openapi

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove groups endpoints from users openapi

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* move constant from postgres to groups

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* move constant from postgres to groups

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* move constant from postgres to groups

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove testing group

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* renam typ to groupType

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* add error for max level

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove print

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove groups.Member interface

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix query building and add test cases

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* uncomment tests

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* move groups package

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove group type, add bulk assign and unassign

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* update openapi, remove parentID from create request, reorder endpoints

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* update openapi

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* update openapi for users and things

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix groups test

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix linter errors

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* resolve comments

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* rename assignReq structure

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* refactor mocks, response, remove type from endpoint

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* some refactor, renaming, errors

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* simplify check

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove package alias

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix naming and comment

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* additional comments

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* add members grpc endpoint test

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix retrieving members for different types

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix retrieving members for different types

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove unecessary structure

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix api grpc

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* rename const

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* refactore retrieve parents and children with common function

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* small changes for errors

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix compile error

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix sorting in mock

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove regexp for groups

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* revert as change is made by mistake

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* revert as change is made by mistake

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* refactor groups and keys package

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix naming

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix naming

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix test for timestamp compare

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix error handling

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* remove errors not being used

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* var renaming

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* resolve comments

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* minor changes

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix test

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* add endpoints for groups into nginx

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* reorganize endpoints, remove some errors

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* reorganize endpoints, remove some errors

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* small fix

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix linter errors

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* minor changes

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* resolve comments

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix group save path problem

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* description constant

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* rename variables

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix validation

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* get back return

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>

* fix compile

Signed-off-by: Mirko Teodorovic <mirko.teodorovic@gmail.com>
2021-03-04 10:29:03 +01:00

779 lines
19 KiB
Go

// Copyright (c) Mainflux
// SPDX-License-Identifier: Apache-2.0
package postgres_test
import (
"context"
"encoding/json"
"fmt"
"strings"
"testing"
"github.com/mainflux/mainflux/pkg/errors"
"github.com/mainflux/mainflux/pkg/uuid"
"github.com/mainflux/mainflux/things"
"github.com/mainflux/mainflux/things/postgres"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
const maxNameSize = 1024
var (
invalidName = strings.Repeat("m", maxNameSize+1)
idProvider = uuid.New()
)
func TestThingsSave(t *testing.T) {
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
email := "thing-save@example.com"
nonexistentThingKey, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
ths := []things.Thing{}
for i := 1; i <= 5; i++ {
thID, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thing := things.Thing{
ID: thID,
Owner: email,
Key: thkey,
}
ths = append(ths, thing)
}
thkey := ths[0].Key
thID := ths[0].ID
cases := []struct {
desc string
things []things.Thing
err error
}{
{
desc: "create new things",
things: ths,
err: nil,
},
{
desc: "create things that already exist",
things: ths,
err: things.ErrConflict,
},
{
desc: "create thing with invalid ID",
things: []things.Thing{
{ID: "invalid", Owner: email, Key: thkey},
},
err: things.ErrMalformedEntity,
},
{
desc: "create thing with invalid name",
things: []things.Thing{
{ID: thID, Owner: email, Key: thkey, Name: invalidName},
},
err: things.ErrMalformedEntity,
},
{
desc: "create thing with invalid Key",
things: []things.Thing{
{ID: thID, Owner: email, Key: nonexistentThingKey},
},
err: things.ErrConflict,
},
{
desc: "create things with conflicting keys",
things: ths,
err: things.ErrConflict,
},
}
for _, tc := range cases {
_, err := thingRepo.Save(context.Background(), tc.things...)
assert.True(t, errors.Contains(err, tc.err), fmt.Sprintf("%s: expected %s got %s\n", tc.desc, tc.err, err))
}
}
func TestThingUpdate(t *testing.T) {
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
email := "thing-update@example.com"
validName := "mfx_device"
thID, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thing := things.Thing{
ID: thID,
Owner: email,
Key: thkey,
}
sths, err := thingRepo.Save(context.Background(), thing)
require.Nil(t, err, fmt.Sprintf("unexpected error: %s", err))
thing.ID = sths[0].ID
nonexistentThingID, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
cases := []struct {
desc string
thing things.Thing
err error
}{
{
desc: "update existing thing",
thing: thing,
err: nil,
},
{
desc: "update non-existing thing with existing user",
thing: things.Thing{
ID: nonexistentThingID,
Owner: email,
},
err: things.ErrNotFound,
},
{
desc: "update existing thing ID with non-existing user",
thing: things.Thing{
ID: thing.ID,
Owner: wrongValue,
},
err: things.ErrNotFound,
},
{
desc: "update non-existing thing with non-existing user",
thing: things.Thing{
ID: nonexistentThingID,
Owner: wrongValue,
},
err: things.ErrNotFound,
},
{
desc: "update thing with valid name",
thing: things.Thing{
ID: thID,
Owner: email,
Key: thkey,
Name: validName,
},
err: nil,
},
{
desc: "update thing with invalid name",
thing: things.Thing{
ID: thID,
Owner: email,
Key: thkey,
Name: invalidName,
},
err: things.ErrMalformedEntity,
},
}
for _, tc := range cases {
err := thingRepo.Update(context.Background(), tc.thing)
assert.True(t, errors.Contains(err, tc.err), fmt.Sprintf("%s: expected %s got %s\n", tc.desc, tc.err, err))
}
}
func TestUpdateKey(t *testing.T) {
email := "thing-update=key@example.com"
newKey := "new-key"
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
id, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
key, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
th1 := things.Thing{
ID: id,
Owner: email,
Key: key,
}
ths, err := thingRepo.Save(context.Background(), th1)
require.Nil(t, err, fmt.Sprintf("unexpected error: %s\n", err))
th1.ID = ths[0].ID
id, err = idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
key, err = idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
th2 := things.Thing{
ID: id,
Owner: email,
Key: key,
}
ths, err = thingRepo.Save(context.Background(), th2)
require.Nil(t, err, fmt.Sprintf("unexpected error: %s\n", err))
th2.ID = ths[0].ID
nonexistentThingID, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
cases := []struct {
desc string
owner string
id string
key string
err error
}{
{
desc: "update key of an existing thing",
owner: th2.Owner,
id: th2.ID,
key: newKey,
err: nil,
},
{
desc: "update key of a non-existing thing with existing user",
owner: th2.Owner,
id: nonexistentThingID,
key: newKey,
err: things.ErrNotFound,
},
{
desc: "update key of an existing thing with non-existing user",
owner: wrongValue,
id: th2.ID,
key: newKey,
err: things.ErrNotFound,
},
{
desc: "update key of a non-existing thing with non-existing user",
owner: wrongValue,
id: nonexistentThingID,
key: newKey,
err: things.ErrNotFound,
},
{
desc: "update key with existing key value",
owner: th2.Owner,
id: th2.ID,
key: th1.Key,
err: things.ErrConflict,
},
}
for _, tc := range cases {
err := thingRepo.UpdateKey(context.Background(), tc.owner, tc.id, tc.key)
assert.True(t, errors.Contains(err, tc.err), fmt.Sprintf("%s: expected %s got %s\n", tc.desc, tc.err, err))
}
}
func TestSingleThingRetrieval(t *testing.T) {
email := "thing-single-retrieval@example.com"
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
id, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
key, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
th := things.Thing{
ID: id,
Owner: email,
Key: key,
}
ths, err := thingRepo.Save(context.Background(), th)
require.Nil(t, err, fmt.Sprintf("unexpected error: %s\n", err))
th.ID = ths[0].ID
nonexistentThingID, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
cases := map[string]struct {
owner string
ID string
err error
}{
"retrieve thing with existing user": {
owner: th.Owner,
ID: th.ID,
err: nil,
},
"retrieve non-existing thing with existing user": {
owner: th.Owner,
ID: nonexistentThingID,
err: things.ErrNotFound,
},
"retrieve thing with non-existing owner": {
owner: wrongValue,
ID: th.ID,
err: things.ErrNotFound,
},
"retrieve thing with malformed ID": {
owner: th.Owner,
ID: wrongValue,
err: things.ErrNotFound,
},
}
for desc, tc := range cases {
_, err := thingRepo.RetrieveByID(context.Background(), tc.owner, tc.ID)
assert.True(t, errors.Contains(err, tc.err), fmt.Sprintf("%s: expected %s got %s\n", desc, tc.err, err))
}
}
func TestThingRetrieveByKey(t *testing.T) {
email := "thing-retrieved-by-key@example.com"
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
id, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
key, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
th := things.Thing{
ID: id,
Owner: email,
Key: key,
}
ths, err := thingRepo.Save(context.Background(), th)
require.Nil(t, err, fmt.Sprintf("unexpected error: %s\n", err))
th.ID = ths[0].ID
cases := map[string]struct {
key string
ID string
err error
}{
"retrieve existing thing by key": {
key: th.Key,
ID: th.ID,
err: nil,
},
"retrieve non-existent thing by key": {
key: wrongValue,
ID: "",
err: things.ErrNotFound,
},
}
for desc, tc := range cases {
id, err := thingRepo.RetrieveByKey(context.Background(), tc.key)
assert.Equal(t, tc.ID, id, fmt.Sprintf("%s: expected %s got %s\n", desc, tc.ID, id))
assert.True(t, errors.Contains(err, tc.err), fmt.Sprintf("%s: expected %s got %s\n", desc, tc.err, err))
}
}
func TestMultiThingRetrieval(t *testing.T) {
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
email := "thing-multi-retrieval@example.com"
name := "thing_name"
metaStr := `{"field1":"value1","field2":{"subfield11":"value2","subfield12":{"subfield121":"value3","subfield122":"value4"}}}`
subMetaStr := `{"field2":{"subfield12":{"subfield121":"value3"}}}`
metadata := things.Metadata{}
json.Unmarshal([]byte(metaStr), &metadata)
subMeta := things.Metadata{}
json.Unmarshal([]byte(subMetaStr), &subMeta)
wrongMeta := things.Metadata{
"field": "value1",
}
offset := uint64(1)
nameNum := uint64(3)
metaNum := uint64(3)
nameMetaNum := uint64(2)
n := uint64(10)
for i := uint64(0); i < n; i++ {
id, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
key, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
th := things.Thing{
Owner: email,
ID: id,
Key: key,
}
// Create Things with name.
if i < nameNum {
th.Name = fmt.Sprintf("%s-%d", name, i)
}
// Create Things with metadata.
if i >= nameNum && i < nameNum+metaNum {
th.Metadata = metadata
}
// Create Things with name and metadata.
if i >= n-nameMetaNum {
th.Metadata = metadata
th.Name = name
}
_, err = thingRepo.Save(context.Background(), th)
require.Nil(t, err, fmt.Sprintf("unexpected error: %s\n", err))
}
cases := map[string]struct {
owner string
pageMetadata things.PageMetadata
size uint64
}{
"retrieve all things with existing owner": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Total: n,
},
size: n,
},
"retrieve subset of things with existing owner": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: n / 2,
Limit: n,
Total: n,
},
size: n / 2,
},
"retrieve things with non-existing owner": {
owner: wrongValue,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Total: 0,
},
size: 0,
},
"retrieve things with existing name": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 1,
Limit: n,
Name: name,
Total: nameNum + nameMetaNum,
},
size: nameNum + nameMetaNum - offset,
},
"retrieve things with non-existing name": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Name: "wrong",
Total: 0,
},
size: 0,
},
"retrieve things with existing metadata": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Total: metaNum + nameMetaNum,
Metadata: metadata,
},
size: metaNum + nameMetaNum,
},
"retrieve things with partial metadata": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Total: metaNum + nameMetaNum,
Metadata: subMeta,
},
size: metaNum + nameMetaNum,
},
"retrieve things with non-existing metadata": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Total: 0,
Metadata: wrongMeta,
},
size: 0,
},
"retrieve all things with existing name and metadata": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Total: nameMetaNum,
Name: name,
Metadata: metadata,
},
size: nameMetaNum,
},
"retrieve things sorted by name ascendent": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Total: n,
Order: "name",
Dir: "asc",
},
size: n,
},
"retrieve things sorted by name descendent": {
owner: email,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Total: n,
Order: "name",
Dir: "desc",
},
size: n,
},
}
for desc, tc := range cases {
page, err := thingRepo.RetrieveAll(context.Background(), tc.owner, tc.pageMetadata)
size := uint64(len(page.Things))
assert.Equal(t, tc.size, size, fmt.Sprintf("%s: expected size %d got %d\n", desc, tc.size, size))
assert.Equal(t, tc.pageMetadata.Total, page.Total, fmt.Sprintf("%s: expected total %d got %d\n", desc, tc.pageMetadata.Total, page.Total))
assert.Nil(t, err, fmt.Sprintf("%s: expected no error got %d\n", desc, err))
// Check if Things list have been sorted properly
testSortThings(t, tc.pageMetadata, page.Things)
}
}
func TestMultiThingRetrievalByChannel(t *testing.T) {
email := "thing-multi-retrieval-by-channel@example.com"
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
channelRepo := postgres.NewChannelRepository(dbMiddleware)
n := uint64(10)
thsDisconNum := uint64(1)
chID, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
_, err = channelRepo.Save(context.Background(), things.Channel{
ID: chID,
Owner: email,
})
require.Nil(t, err, fmt.Sprintf("unexpected error: %s", err))
for i := uint64(0); i < n; i++ {
thID, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
th := things.Thing{
ID: thID,
Owner: email,
Key: thkey,
}
ths, err := thingRepo.Save(context.Background(), th)
require.Nil(t, err, fmt.Sprintf("unexpected error: %s", err))
thID = ths[0].ID
// Don't connnect last Thing
if i == n-thsDisconNum {
break
}
err = channelRepo.Connect(context.Background(), email, []string{chID}, []string{thID})
require.Nil(t, err, fmt.Sprintf("unexpected error: %s", err))
}
nonexistentChanID, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
cases := map[string]struct {
owner string
chID string
pageMetadata things.PageMetadata
size uint64
err error
}{
"retrieve all things by channel with existing owner": {
owner: email,
chID: chID,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: true,
},
size: n - thsDisconNum,
},
"retrieve subset of things by channel with existing owner": {
owner: email,
chID: chID,
pageMetadata: things.PageMetadata{
Offset: n / 2,
Limit: n,
Connected: true,
},
size: (n / 2) - thsDisconNum,
},
"retrieve things by channel with non-existing owner": {
owner: wrongValue,
chID: chID,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: true,
},
size: 0,
},
"retrieve things by non-existing channel": {
owner: email,
chID: nonexistentChanID,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: true,
},
size: 0,
},
"retrieve things with malformed UUID": {
owner: email,
chID: wrongValue,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: true,
},
size: 0,
err: things.ErrNotFound,
},
"retrieve all non connected things by channel with existing owner": {
owner: email,
chID: chID,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: false,
},
size: thsDisconNum,
},
"retrieve all things by channel sorted by name ascendent": {
owner: email,
chID: chID,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: true,
Order: "name",
Dir: "asc",
},
size: n - thsDisconNum,
},
"retrieve all non-connected things by channel sorted by name ascendent": {
owner: email,
chID: chID,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: false,
Order: "name",
Dir: "asc",
},
size: thsDisconNum,
},
"retrieve all things by channel sorted by name descendent": {
owner: email,
chID: chID,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: true,
Order: "name",
Dir: "desc",
},
size: n - thsDisconNum,
},
"retrieve all non-connected things by channel sorted by name descendent": {
owner: email,
chID: chID,
pageMetadata: things.PageMetadata{
Offset: 0,
Limit: n,
Connected: false,
Order: "name",
Dir: "desc",
},
size: thsDisconNum,
},
}
for desc, tc := range cases {
page, err := thingRepo.RetrieveByChannel(context.Background(), tc.owner, tc.chID, tc.pageMetadata)
size := uint64(len(page.Things))
assert.Equal(t, tc.size, size, fmt.Sprintf("%s: expected size %d got %d\n", desc, tc.size, size))
assert.True(t, errors.Contains(err, tc.err), fmt.Sprintf("%s: expected no error got %d\n", desc, err))
// Check if Things by Channel list have been sorted properly
testSortThings(t, tc.pageMetadata, page.Things)
}
}
func TestThingRemoval(t *testing.T) {
email := "thing-removal@example.com"
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
id, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
key, err := idProvider.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thing := things.Thing{
ID: id,
Owner: email,
Key: key,
}
ths, _ := thingRepo.Save(context.Background(), thing)
thing.ID = ths[0].ID
// show that the removal works the same for both existing and non-existing
// (removed) thing
for i := 0; i < 2; i++ {
err := thingRepo.Remove(context.Background(), email, thing.ID)
require.Nil(t, err, fmt.Sprintf("#%d: failed to remove thing due to: %s", i, err))
_, err = thingRepo.RetrieveByID(context.Background(), email, thing.ID)
require.True(t, errors.Contains(err, things.ErrNotFound), fmt.Sprintf("#%d: expected %s got %s", i, things.ErrNotFound, err))
}
}
func testSortThings(t *testing.T, pm things.PageMetadata, ths []things.Thing) {
switch pm.Order {
case "name":
current := ths[0]
for _, res := range ths {
if pm.Dir == "asc" {
assert.GreaterOrEqual(t, res.Name, current.Name)
}
if pm.Dir == "desc" {
assert.GreaterOrEqual(t, current.Name, res.Name)
}
current = res
}
default:
break
}
}