1
0
mirror of https://github.com/mainflux/mainflux.git synced 2025-04-24 13:48:49 +08:00
Ivan Milošević d2153a8846
[NOISSUE] - Improve errors package (#1086)
* Init commit - implement errors package on things service

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* things service errors issue

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* Decode errors
Add authn service to run script

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* Modify tests

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* service_test

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* debug lines

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* Regulate tests

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* Improve errors in Redis

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* Remove dead code
Inline if conditions
Rename err var

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* Transform errors messages to lowercase

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* improve errors package

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* modify wrap method
inline wrapping errors in redis

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* Add copyright to errors package

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* wrapping nil error returns wrapper (instead of nil)

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* move response messages in test to vars

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* golangcibot review fix

Signed-off-by: Ivan Milošević <iva@blokovi.com>

* golangbot fix review in transport

Signed-off-by: Ivan Milošević <iva@blokovi.com>
2020-03-30 15:22:18 +02:00

636 lines
15 KiB
Go

// Copyright (c) Mainflux
// SPDX-License-Identifier: Apache-2.0
package postgres_test
import (
"context"
"fmt"
"strings"
"testing"
"github.com/mainflux/mainflux/errors"
"github.com/mainflux/mainflux/things"
"github.com/mainflux/mainflux/things/postgres"
"github.com/mainflux/mainflux/things/uuid"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
const maxNameSize = 1024
var invalidName = strings.Repeat("m", maxNameSize+1)
func TestThingsSave(t *testing.T) {
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
email := "thing-save@example.com"
nonexistentThingKey, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
var thid string
var thkey string
ths := []things.Thing{}
for i := 1; i <= 5; i++ {
thid, err = uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err = uuid.New().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)
}
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{
things.Thing{
ID: "invalid",
Owner: email,
Key: thkey,
},
},
err: things.ErrMalformedEntity,
},
{
desc: "create thing with invalid name",
things: []things.Thing{
things.Thing{
ID: thid,
Owner: email,
Key: thkey,
Name: invalidName,
},
},
err: things.ErrMalformedEntity,
},
{
desc: "create thing with invalid Key",
things: []things.Thing{
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 := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thing := things.Thing{
ID: thid,
Owner: email,
Key: thkey,
}
sths, _ := thingRepo.Save(context.Background(), thing)
thing.ID = sths[0].ID
nonexistentThingID, err := uuid.New().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)
ethid, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
ethkey, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
existingThing := things.Thing{
ID: ethid,
Owner: email,
Key: ethkey,
}
sths, _ := thingRepo.Save(context.Background(), existingThing)
existingThing.ID = sths[0].ID
thid, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thing := things.Thing{
ID: thid,
Owner: email,
Key: thkey,
}
sths, _ = thingRepo.Save(context.Background(), thing)
thing.ID = sths[0].ID
nonexistentThingID, err := uuid.New().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: thing.Owner,
id: thing.ID,
key: newKey,
err: nil,
},
{
desc: "update key of a non-existing thing with existing user",
owner: thing.Owner,
id: nonexistentThingID,
key: newKey,
err: things.ErrNotFound,
},
{
desc: "update key of an existing thing with non-existing user",
owner: wrongValue,
id: thing.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: thing.Owner,
id: thing.ID,
key: existingThing.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)
thid, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thing := things.Thing{
ID: thid,
Owner: email,
Key: thkey,
}
sths, _ := thingRepo.Save(context.Background(), thing)
thing.ID = sths[0].ID
nonexistentThingID, err := uuid.New().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: thing.Owner,
ID: thing.ID,
err: nil,
},
"retrieve non-existing thing with existing user": {
owner: thing.Owner,
ID: nonexistentThingID,
err: things.ErrNotFound,
},
"retrieve thing with non-existing owner": {
owner: wrongValue,
ID: thing.ID,
err: things.ErrNotFound,
},
"retrieve thing with malformed ID": {
owner: thing.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)
thid, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thing := things.Thing{
ID: thid,
Owner: email,
Key: thkey,
}
sths, _ := thingRepo.Save(context.Background(), thing)
thing.ID = sths[0].ID
cases := map[string]struct {
key string
ID string
err error
}{
"retrieve existing thing by key": {
key: thing.Key,
ID: thing.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 := "mainflux"
metadata := things.Metadata{
"field": "value",
}
wrongMeta := things.Metadata{
"wrong": "wrong",
}
idp := uuid.New()
offset := uint64(1)
thNameNum := uint64(3)
thMetaNum := uint64(3)
thNameMetaNum := uint64(2)
n := uint64(10)
for i := uint64(0); i < n; i++ {
thid, err := idp.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := idp.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
th := things.Thing{
Owner: email,
ID: thid,
Key: thkey,
}
// Create Things with name.
if i < thNameNum {
th.Name = name
}
// Create Things with metadata.
if i >= thNameNum && i < thNameNum+thMetaNum {
th.Metadata = metadata
}
// Create Things with name and metadata.
if i >= n-thNameMetaNum {
th.Metadata = metadata
th.Name = name
}
thingRepo.Save(context.Background(), th)
}
cases := map[string]struct {
owner string
offset uint64
limit uint64
name string
size uint64
total uint64
metadata map[string]interface{}
}{
"retrieve all things with existing owner": {
owner: email,
offset: 0,
limit: n,
size: n,
total: n,
},
"retrieve subset of things with existing owner": {
owner: email,
offset: n / 2,
limit: n,
size: n / 2,
total: n,
},
"retrieve things with non-existing owner": {
owner: wrongValue,
offset: 0,
limit: n,
size: 0,
total: 0,
},
"retrieve things with existing name": {
owner: email,
offset: 1,
limit: n,
name: name,
size: thNameNum + thNameMetaNum - offset,
total: thNameNum + thNameMetaNum,
},
"retrieve things with non-existing name": {
owner: email,
offset: 0,
limit: n,
name: "wrong",
size: 0,
total: 0,
},
"retrieve things with existing metadata": {
owner: email,
offset: 0,
limit: n,
size: thMetaNum + thNameMetaNum,
total: thMetaNum + thNameMetaNum,
metadata: metadata,
},
"retrieve things with non-existing metadata": {
owner: email,
offset: 0,
limit: n,
size: 0,
total: 0,
metadata: wrongMeta,
},
"retrieve all things with existing name and metadata": {
owner: email,
offset: 0,
limit: n,
size: thNameMetaNum,
total: thNameMetaNum,
name: name,
metadata: metadata,
},
}
for desc, tc := range cases {
page, err := thingRepo.RetrieveAll(context.Background(), tc.owner, tc.offset, tc.limit, tc.name, tc.metadata)
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.total, page.Total, fmt.Sprintf("%s: expected total %d got %d\n", desc, tc.total, page.Total))
assert.Nil(t, err, fmt.Sprintf("%s: expected no error got %d\n", desc, err))
}
}
func TestMultiThingRetrievalByChannel(t *testing.T) {
email := "thing-multi-retrieval-by-channel@example.com"
idp := uuid.New()
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
channelRepo := postgres.NewChannelRepository(dbMiddleware)
n := uint64(10)
chid, err := idp.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
schs, err := channelRepo.Save(context.Background(), things.Channel{
ID: chid,
Owner: email,
})
require.Nil(t, err, fmt.Sprintf("unexpected error: %s", err))
cid := schs[0].ID
for i := uint64(0); i < n; i++ {
thid, err := idp.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := idp.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
th := things.Thing{
ID: thid,
Owner: email,
Key: thkey,
}
sths, err := thingRepo.Save(context.Background(), th)
require.Nil(t, err, fmt.Sprintf("unexpected error: %s", err))
tid := sths[0].ID
err = channelRepo.Connect(context.Background(), email, []string{cid}, []string{tid})
require.Nil(t, err, fmt.Sprintf("unexpected error: %s", err))
}
nonexistentChanID, err := idp.ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
cases := map[string]struct {
owner string
channel string
offset uint64
limit uint64
size uint64
err error
}{
"retrieve all things by channel with existing owner": {
owner: email,
channel: cid,
offset: 0,
limit: n,
size: n,
},
"retrieve subset of things by channel with existing owner": {
owner: email,
channel: cid,
offset: n / 2,
limit: n,
size: n / 2,
},
"retrieve things by channel with non-existing owner": {
owner: wrongValue,
channel: cid,
offset: 0,
limit: n,
size: 0,
},
"retrieve things by non-existing channel": {
owner: email,
channel: nonexistentChanID,
offset: 0,
limit: n,
size: 0,
},
"retrieve things with malformed UUID": {
owner: email,
channel: wrongValue,
offset: 0,
limit: n,
size: 0,
err: things.ErrNotFound,
},
}
for desc, tc := range cases {
page, err := thingRepo.RetrieveByChannel(context.Background(), tc.owner, tc.channel, tc.offset, tc.limit)
size := uint64(len(page.Things))
assert.Equal(t, tc.size, size, fmt.Sprintf("%s: expected %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))
}
}
func TestThingRemoval(t *testing.T) {
email := "thing-removal@example.com"
dbMiddleware := postgres.NewDatabase(db)
thingRepo := postgres.NewThingRepository(dbMiddleware)
thid, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thkey, err := uuid.New().ID()
require.Nil(t, err, fmt.Sprintf("got unexpected error: %s", err))
thing := things.Thing{
ID: thid,
Owner: email,
Key: thkey,
}
sths, _ := thingRepo.Save(context.Background(), thing)
thing.ID = sths[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.Equal(t, things.ErrNotFound, err, fmt.Sprintf("#%d: expected %s got %s", i, things.ErrNotFound, err))
require.True(t, errors.Contains(err, things.ErrNotFound), fmt.Sprintf("#%d: expected %s got %s", i, things.ErrNotFound, err))
}
}