2020-01-09 01:21:09 +01:00
|
|
|
package classify
|
2018-09-14 12:44:15 +02:00
|
|
|
|
|
|
|
import (
|
2019-04-29 20:09:10 +02:00
|
|
|
"io/ioutil"
|
2020-04-30 16:11:27 +02:00
|
|
|
"sync"
|
2018-09-14 12:44:15 +02:00
|
|
|
"testing"
|
2018-10-31 07:14:33 +01:00
|
|
|
|
2020-05-31 02:09:52 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
2019-12-11 14:10:20 +01:00
|
|
|
tensorflow "github.com/tensorflow/tensorflow/tensorflow/go"
|
|
|
|
|
2018-10-31 07:14:33 +01:00
|
|
|
"github.com/stretchr/testify/assert"
|
2018-09-14 12:44:15 +02:00
|
|
|
)
|
|
|
|
|
2020-05-31 02:09:52 +02:00
|
|
|
var assetsPath = fs.Abs("../../assets")
|
|
|
|
var modelPath = assetsPath + "/nasnet"
|
|
|
|
var examplesPath = assetsPath + "/examples"
|
2020-04-30 16:11:27 +02:00
|
|
|
var once sync.Once
|
|
|
|
var testInstance *TensorFlow
|
|
|
|
|
|
|
|
// NewTest returns a new TensorFlow test instance.
|
|
|
|
func NewTest(t *testing.T) *TensorFlow {
|
|
|
|
once.Do(func() {
|
2020-05-31 02:09:52 +02:00
|
|
|
testInstance = New(assetsPath, false)
|
2020-04-30 16:11:27 +02:00
|
|
|
if err := testInstance.loadModel(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-04-29 11:18:26 +02:00
|
|
|
})
|
|
|
|
|
2020-04-30 16:11:27 +02:00
|
|
|
return testInstance
|
2020-04-29 11:18:26 +02:00
|
|
|
}
|
|
|
|
|
2019-06-04 18:26:35 +02:00
|
|
|
func TestTensorFlow_LabelsFromFile(t *testing.T) {
|
2020-01-09 01:21:09 +01:00
|
|
|
t.Run("chameleon_lime.jpg", func(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
tensorFlow := NewTest(t)
|
2018-09-14 12:44:15 +02:00
|
|
|
|
2020-04-16 20:57:00 +02:00
|
|
|
result, err := tensorFlow.File(examplesPath + "/chameleon_lime.jpg")
|
2018-09-14 12:44:15 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.Nil(t, err)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
if err != nil {
|
2020-04-30 16:11:27 +02:00
|
|
|
t.Fatal(err)
|
2019-07-17 10:48:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
assert.NotNil(t, result)
|
|
|
|
assert.IsType(t, Labels{}, result)
|
|
|
|
assert.Equal(t, 1, len(result))
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
t.Log(result)
|
|
|
|
|
|
|
|
assert.Equal(t, "chameleon", result[0].Name)
|
2018-09-14 12:44:15 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.Equal(t, 7, result[0].Uncertainty)
|
|
|
|
})
|
|
|
|
t.Run("not existing file", func(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
tensorFlow := NewTest(t)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2020-04-16 20:57:00 +02:00
|
|
|
result, err := tensorFlow.File(examplesPath + "/notexisting.jpg")
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.Contains(t, err.Error(), "no such file or directory")
|
|
|
|
assert.Empty(t, result)
|
|
|
|
})
|
2020-04-29 11:18:26 +02:00
|
|
|
t.Run("disabled true", func(t *testing.T) {
|
2020-05-31 02:09:52 +02:00
|
|
|
tensorFlow := New(assetsPath, true)
|
2020-04-29 11:18:26 +02:00
|
|
|
|
|
|
|
result, err := tensorFlow.File(examplesPath + "/chameleon_lime.jpg")
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
if err != nil {
|
2020-04-30 16:11:27 +02:00
|
|
|
t.Fatal(err)
|
2020-04-29 11:18:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
assert.Nil(t, result)
|
|
|
|
assert.IsType(t, Labels{}, result)
|
|
|
|
assert.Equal(t, 0, len(result))
|
|
|
|
|
|
|
|
t.Log(result)
|
|
|
|
})
|
2018-09-14 12:44:15 +02:00
|
|
|
}
|
2019-04-29 20:09:10 +02:00
|
|
|
|
2019-06-04 18:26:35 +02:00
|
|
|
func TestTensorFlow_Labels(t *testing.T) {
|
2019-04-29 20:09:10 +02:00
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skipping test in short mode.")
|
|
|
|
}
|
|
|
|
|
2020-01-09 01:21:09 +01:00
|
|
|
t.Run("chameleon_lime.jpg", func(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
tensorFlow := NewTest(t)
|
2019-04-29 20:09:10 +02:00
|
|
|
|
2020-04-16 20:57:00 +02:00
|
|
|
if imageBuffer, err := ioutil.ReadFile(examplesPath + "/chameleon_lime.jpg"); err != nil {
|
2019-07-17 10:48:23 +02:00
|
|
|
t.Error(err)
|
|
|
|
} else {
|
|
|
|
result, err := tensorFlow.Labels(imageBuffer)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
t.Log(result)
|
2019-04-29 20:09:10 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.NotNil(t, result)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2020-04-30 16:11:27 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.IsType(t, Labels{}, result)
|
|
|
|
assert.Equal(t, 1, len(result))
|
2019-04-29 20:09:10 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.Equal(t, "chameleon", result[0].Name)
|
2019-04-29 20:09:10 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.Equal(t, 100-93, result[0].Uncertainty)
|
|
|
|
}
|
|
|
|
})
|
2020-01-09 01:21:09 +01:00
|
|
|
t.Run("dog_orange.jpg", func(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
tensorFlow := NewTest(t)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2020-04-16 20:57:00 +02:00
|
|
|
if imageBuffer, err := ioutil.ReadFile(examplesPath + "/dog_orange.jpg"); err != nil {
|
2019-07-17 10:48:23 +02:00
|
|
|
t.Error(err)
|
|
|
|
} else {
|
|
|
|
result, err := tensorFlow.Labels(imageBuffer)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
t.Log(result)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.NotNil(t, result)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2020-04-30 16:11:27 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.IsType(t, Labels{}, result)
|
2019-12-16 20:22:46 +01:00
|
|
|
assert.Equal(t, 1, len(result))
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2020-04-27 15:45:43 +02:00
|
|
|
assert.Equal(t, "dog", result[0].Name)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.Equal(t, 34, result[0].Uncertainty)
|
|
|
|
}
|
|
|
|
})
|
2020-01-09 01:21:09 +01:00
|
|
|
t.Run("Random.docx", func(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
tensorFlow := NewTest(t)
|
2019-04-30 13:17:01 +02:00
|
|
|
|
2020-04-16 20:57:00 +02:00
|
|
|
if imageBuffer, err := ioutil.ReadFile(examplesPath + "/Random.docx"); err != nil {
|
2019-07-17 10:48:23 +02:00
|
|
|
t.Error(err)
|
|
|
|
} else {
|
|
|
|
result, err := tensorFlow.Labels(imageBuffer)
|
|
|
|
assert.Empty(t, result)
|
2020-04-30 16:11:27 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
assert.Contains(t, err.Error(), "invalid image")
|
|
|
|
} else {
|
|
|
|
t.Fatal("err should NOT be nil")
|
|
|
|
}
|
2019-07-17 10:48:23 +02:00
|
|
|
}
|
|
|
|
})
|
2020-01-09 01:21:09 +01:00
|
|
|
t.Run("6720px_white.jpg", func(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
tensorFlow := NewTest(t)
|
2019-07-17 10:48:23 +02:00
|
|
|
|
2020-04-16 20:57:00 +02:00
|
|
|
if imageBuffer, err := ioutil.ReadFile(examplesPath + "/6720px_white.jpg"); err != nil {
|
2019-07-17 10:48:23 +02:00
|
|
|
t.Error(err)
|
|
|
|
} else {
|
|
|
|
result, err := tensorFlow.Labels(imageBuffer)
|
2020-04-30 16:11:27 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2019-07-17 10:48:23 +02:00
|
|
|
assert.Empty(t, result)
|
|
|
|
}
|
|
|
|
})
|
2020-04-29 11:18:26 +02:00
|
|
|
t.Run("disabled true", func(t *testing.T) {
|
2020-05-31 02:09:52 +02:00
|
|
|
tensorFlow := New(assetsPath, true)
|
2020-04-29 11:18:26 +02:00
|
|
|
|
|
|
|
if imageBuffer, err := ioutil.ReadFile(examplesPath + "/dog_orange.jpg"); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
} else {
|
|
|
|
result, err := tensorFlow.Labels(imageBuffer)
|
|
|
|
|
|
|
|
t.Log(result)
|
|
|
|
|
|
|
|
assert.Nil(t, result)
|
|
|
|
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.IsType(t, Labels{}, result)
|
|
|
|
assert.Equal(t, 0, len(result))
|
|
|
|
}
|
|
|
|
})
|
2019-07-17 10:48:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestTensorFlow_LoadModel(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
t.Run("model loaded", func(t *testing.T) {
|
|
|
|
tf := NewTest(t)
|
|
|
|
assert.True(t, tf.ModelLoaded())
|
2019-07-17 10:48:23 +02:00
|
|
|
})
|
|
|
|
t.Run("model path does not exist", func(t *testing.T) {
|
2020-05-31 02:09:52 +02:00
|
|
|
tensorFlow := New(assetsPath+"foo", false)
|
2020-04-30 16:11:27 +02:00
|
|
|
if err := tensorFlow.loadModel(); err != nil {
|
|
|
|
assert.Contains(t, err.Error(), "Could not find SavedModel")
|
|
|
|
} else {
|
|
|
|
t.Fatal("err should NOT be nil")
|
2020-04-16 20:57:00 +02:00
|
|
|
}
|
2019-07-17 10:48:23 +02:00
|
|
|
})
|
2019-04-29 20:09:10 +02:00
|
|
|
}
|
2019-07-17 10:59:19 +02:00
|
|
|
|
2019-07-17 11:52:26 +02:00
|
|
|
func TestTensorFlow_BestLabels(t *testing.T) {
|
|
|
|
t.Run("labels not loaded", func(t *testing.T) {
|
2020-05-31 02:09:52 +02:00
|
|
|
tensorFlow := New(assetsPath, false)
|
2019-07-17 11:52:26 +02:00
|
|
|
|
|
|
|
p := make([]float32, 1000)
|
|
|
|
|
|
|
|
p[666] = 0.5
|
|
|
|
|
|
|
|
result := tensorFlow.bestLabels(p)
|
|
|
|
assert.Empty(t, result)
|
|
|
|
})
|
2019-07-17 12:22:50 +02:00
|
|
|
t.Run("labels loaded", func(t *testing.T) {
|
2020-05-31 02:09:52 +02:00
|
|
|
tensorFlow := New(assetsPath, false)
|
2020-04-30 16:11:27 +02:00
|
|
|
|
|
|
|
if err := tensorFlow.loadLabels(modelPath); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-07-17 11:52:26 +02:00
|
|
|
|
|
|
|
p := make([]float32, 1000)
|
|
|
|
|
|
|
|
p[8] = 0.7
|
|
|
|
p[1] = 0.5
|
|
|
|
|
|
|
|
result := tensorFlow.bestLabels(p)
|
2019-12-15 18:36:27 +01:00
|
|
|
assert.Equal(t, "chicken", result[0].Name)
|
2019-07-17 11:52:26 +02:00
|
|
|
assert.Equal(t, "bird", result[0].Categories[0])
|
|
|
|
assert.Equal(t, "image", result[0].Source)
|
|
|
|
t.Log(result)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-07-17 12:22:50 +02:00
|
|
|
func TestTensorFlow_MakeTensor(t *testing.T) {
|
2020-01-09 01:21:09 +01:00
|
|
|
t.Run("cat_brown.jpg", func(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
tensorFlow := NewTest(t)
|
2019-07-17 12:22:50 +02:00
|
|
|
|
2020-04-16 20:57:00 +02:00
|
|
|
imageBuffer, err := ioutil.ReadFile(examplesPath + "/cat_brown.jpg")
|
2020-05-31 02:09:52 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2019-07-17 12:22:50 +02:00
|
|
|
result, err := tensorFlow.makeTensor(imageBuffer, "jpeg")
|
|
|
|
assert.Equal(t, tensorflow.DataType(0x1), result.DataType())
|
|
|
|
assert.Equal(t, int64(1), result.Shape()[0])
|
|
|
|
assert.Equal(t, int64(224), result.Shape()[2])
|
|
|
|
})
|
2020-01-09 01:21:09 +01:00
|
|
|
t.Run("Random.docx", func(t *testing.T) {
|
2020-04-30 16:11:27 +02:00
|
|
|
tensorFlow := NewTest(t)
|
2019-07-17 12:22:50 +02:00
|
|
|
|
2020-04-16 20:57:00 +02:00
|
|
|
imageBuffer, err := ioutil.ReadFile(examplesPath + "/Random.docx")
|
2019-07-17 12:22:50 +02:00
|
|
|
assert.Nil(t, err)
|
|
|
|
result, err := tensorFlow.makeTensor(imageBuffer, "jpeg")
|
|
|
|
assert.Empty(t, result)
|
|
|
|
assert.Equal(t, "image: unknown format", err.Error())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-07-17 11:52:26 +02:00
|
|
|
func Test_ConvertTF(t *testing.T) {
|
|
|
|
result := convertTF(uint32(98765432))
|
|
|
|
assert.Equal(t, float32(3024.898), result)
|
|
|
|
}
|