/**
\file test_SimpleColorHuewStrategy.cpp
Copyright Notice\n
Copyright (C) 2017 Jan Rogall - developer\n
Copyright (C) 2017 Moritz Wirger - developer\n
This file is part of hueplusplus.
hueplusplus is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
hueplusplus is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with hueplusplus. If not, see .
**/
#include
#include
#include
#include
#include "TestTransaction.h"
#include "testhelper.h"
#include "hueplusplus/SimpleColorHueStrategy.h"
#include "json/json.hpp"
#include "mocks/mock_HttpHandler.h"
#include "mocks/mock_Light.h"
using namespace hueplusplus;
TEST(SimpleColorHueStrategy, setColorHue)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight test_light(handler);
const std::string statePath = "/api/" + getBridgeUsername() + "/lights/1/state";
nlohmann::json prep_ret;
prep_ret = nlohmann::json::array();
prep_ret[0] = nlohmann::json::object();
prep_ret[0]["success"] = nlohmann::json::object();
prep_ret[0]["success"]["/lights/1/state/transitiontime"] = 6;
prep_ret[1] = nlohmann::json::object();
prep_ret[1]["success"] = nlohmann::json::object();
prep_ret[1]["success"]["/lights/1/state/on"] = true;
prep_ret[2] = nlohmann::json::object();
prep_ret[2]["success"] = nlohmann::json::object();
prep_ret[2]["success"]["/lights/1/state/hue"] = 30500;
EXPECT_CALL(*handler, PUTJson(statePath, _, getBridgeIp(), getBridgePort())).Times(1).WillOnce(Return(prep_ret));
test_light.getState()["state"]["on"] = true;
test_light.getState()["state"]["hue"] = 200;
test_light.getState()["state"]["colormode"] = "hs";
EXPECT_EQ(true, SimpleColorHueStrategy().setColorHue(200, 4, test_light));
test_light.getState()["state"]["on"] = false;
EXPECT_EQ(true, SimpleColorHueStrategy().setColorHue(30500, 6, test_light));
}
TEST(SimpleColorHueStrategy, setColorSaturation)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight test_light(handler);
const std::string statePath = "/api/" + getBridgeUsername() + "/lights/1/state";
nlohmann::json prep_ret;
prep_ret = nlohmann::json::array();
prep_ret[0] = nlohmann::json::object();
prep_ret[0]["success"] = nlohmann::json::object();
prep_ret[0]["success"]["/lights/1/state/transitiontime"] = 6;
prep_ret[1] = nlohmann::json::object();
prep_ret[1]["success"] = nlohmann::json::object();
prep_ret[1]["success"]["/lights/1/state/on"] = true;
prep_ret[2] = nlohmann::json::object();
prep_ret[2]["success"] = nlohmann::json::object();
prep_ret[2]["success"]["/lights/1/state/sat"] = 254;
EXPECT_CALL(*handler, PUTJson(statePath, _, getBridgeIp(), getBridgePort())).Times(1).WillOnce(Return(prep_ret));
test_light.getState()["state"]["on"] = true;
test_light.getState()["state"]["sat"] = 100;
test_light.getState()["state"]["colormode"] = "hs";
EXPECT_EQ(true, SimpleColorHueStrategy().setColorSaturation(100, 4, test_light));
test_light.getState()["state"]["on"] = false;
EXPECT_EQ(true, SimpleColorHueStrategy().setColorSaturation(255, 6, test_light));
}
TEST(SimpleColorHueStrategy, setColorHueSaturation)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight test_light(handler);
const std::string statePath = "/api/" + getBridgeUsername() + "/lights/1/state";
nlohmann::json prep_ret;
prep_ret = nlohmann::json::array();
prep_ret[0] = nlohmann::json::object();
prep_ret[0]["success"] = nlohmann::json::object();
prep_ret[0]["success"]["/lights/1/state/transitiontime"] = 6;
prep_ret[1] = nlohmann::json::object();
prep_ret[1]["success"] = nlohmann::json::object();
prep_ret[1]["success"]["/lights/1/state/on"] = true;
prep_ret[2] = nlohmann::json::object();
prep_ret[2]["success"] = nlohmann::json::object();
prep_ret[2]["success"]["/lights/1/state/hue"] = 30500;
prep_ret[3] = nlohmann::json::object();
prep_ret[3]["success"] = nlohmann::json::object();
prep_ret[3]["success"]["/lights/1/state/sat"] = 254;
EXPECT_CALL(*handler, PUTJson(statePath, _, getBridgeIp(), getBridgePort())).Times(1).WillOnce(Return(prep_ret));
test_light.getState()["state"]["on"] = true;
test_light.getState()["state"]["sat"] = 100;
test_light.getState()["state"]["hue"] = 200;
test_light.getState()["state"]["colormode"] = "hs";
EXPECT_EQ(true, SimpleColorHueStrategy().setColorHueSaturation({200, 100}, 4, test_light));
test_light.getState()["state"]["on"] = false;
EXPECT_EQ(true, SimpleColorHueStrategy().setColorHueSaturation({30500, 255}, 6, test_light));
}
TEST(SimpleColorHueStrategy, setColorXY)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight test_light(handler);
const std::string statePath = "/api/" + getBridgeUsername() + "/lights/1/state";
nlohmann::json prep_ret
= {{{"success", {{"/lights/1/state/transitiontime", 6}}}}, {{"success", {{"/lights/1/state/on", true}}}},
{{"success", {{"/lights/1/state/xy", {0.2355, 0.1234}}}}}, {{"success", {{"/lights/1/state/bri", 254}}}}};
EXPECT_CALL(*handler, PUTJson(statePath, _, getBridgeIp(), getBridgePort())).Times(1).WillOnce(Return(prep_ret));
test_light.getState()["state"]["on"] = true;
test_light.getState()["state"]["xy"][0] = 0.1f;
test_light.getState()["state"]["xy"][1] = 0.1f;
test_light.getState()["state"]["bri"] = 254;
test_light.getState()["state"]["colormode"] = "xy";
EXPECT_EQ(true, SimpleColorHueStrategy().setColorXY({{0.1f, 0.1f}, 1.f}, 4, test_light));
test_light.getState()["state"]["on"] = false;
test_light.getState()["state"]["bri"] = 0;
EXPECT_EQ(true, SimpleColorHueStrategy().setColorXY({{0.2355f, 0.1234f}, 1.f}, 6, test_light));
}
TEST(SimpleColorHueStrategy, setColorLoop)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight test_light(handler);
const std::string statePath = "/api/" + getBridgeUsername() + "/lights/1/state";
nlohmann::json prep_ret;
prep_ret = nlohmann::json::array();
prep_ret[0] = nlohmann::json::object();
prep_ret[0]["success"] = nlohmann::json::object();
prep_ret[0]["success"]["/lights/1/state/on"] = true;
prep_ret[1] = nlohmann::json::object();
prep_ret[1]["success"] = nlohmann::json::object();
prep_ret[1]["success"]["/lights/1/state/effect"] = "colorloop";
EXPECT_CALL(*handler, PUTJson(statePath, _, getBridgeIp(), getBridgePort())).Times(1).WillOnce(Return(prep_ret));
test_light.getState()["state"]["on"] = true;
test_light.getState()["state"]["effect"] = "colorloop";
EXPECT_EQ(true, SimpleColorHueStrategy().setColorLoop(true, test_light));
test_light.getState()["state"]["on"] = false;
test_light.getState()["state"]["effect"] = "none";
EXPECT_EQ(true, SimpleColorHueStrategy().setColorLoop(true, test_light));
}
TEST(SimpleColorHueStrategy, alertHueSaturation)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight light(handler);
// Invalid colormode
{
light.getState()["state"]["colormode"] = "invalid";
light.getState()["state"]["on"] = false;
EXPECT_EQ(false, SimpleColorHueStrategy().alertHueSaturation({30000, 128}, light));
}
const HueSaturation hueSat {200, 100};
// Needs to update the state so transactions are correctly trimmed
const auto setColorLambda = [&](const HueSaturation& hueSat, int transition) {
light.getState()["state"]["colormode"] = "hs";
light.getState()["state"]["on"] = true;
light.getState()["state"]["hue"] = hueSat.hue;
light.getState()["state"]["sat"] = hueSat.saturation;
return true;
};
// Colormode hs
{
const nlohmann::json state
= {{"colormode", "hs"}, {"on", true}, {"xy", {0.1, 0.1}}, {"hue", 300}, {"sat", 100}, {"bri", 254}};
light.getState()["state"] = state;
EXPECT_CALL(Const(light), getBrightness()).Times(AnyNumber()).WillRepeatedly(Return(254));
EXPECT_CALL(Const(light), getColorHueSaturation())
.Times(AnyNumber())
.WillRepeatedly(Return(HueSaturation {300, 100}));
TestTransaction reverseTransaction = light.transaction().setColorHue(300).setTransition(1);
// Set color fails
{
EXPECT_CALL(light, setColorHueSaturation(hueSat, 1)).WillOnce(Return(false));
EXPECT_FALSE(SimpleColorHueStrategy().alertHueSaturation(hueSat, light));
}
// Alert call fails
{
InSequence s;
EXPECT_CALL(light, setColorHueSaturation(hueSat, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(false));
EXPECT_FALSE(SimpleColorHueStrategy().alertHueSaturation(hueSat, light));
}
light.getState()["state"] = state;
// Reverse transaction fails
{
InSequence s;
EXPECT_CALL(light, setColorHueSaturation(hueSat, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectPut(handler).WillOnce(Return(nlohmann::json::object()));
EXPECT_FALSE(SimpleColorHueStrategy().alertHueSaturation(hueSat, light));
}
light.getState()["state"] = state;
Mock::VerifyAndClearExpectations(handler.get());
// Successful
{
InSequence s;
EXPECT_CALL(light, setColorHueSaturation(hueSat, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectSuccessfulPut(handler, Exactly(1));
EXPECT_TRUE(SimpleColorHueStrategy().alertHueSaturation(hueSat, light));
}
Mock::VerifyAndClearExpectations(handler.get());
}
// Colormode hs, off
{
const nlohmann::json state
= {{"colormode", "hs"}, {"on", false}, {"xy", {0.1, 0.1}}, {"hue", 300}, {"sat", 100}, {"bri", 254}};
light.getState()["state"] = state;
TestTransaction reverseTransaction = light.transaction().setColorHue(300).setOn(false).setTransition(1);
InSequence s;
EXPECT_CALL(light, setColorHueSaturation(hueSat, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectSuccessfulPut(handler, Exactly(1));
EXPECT_TRUE(SimpleColorHueStrategy().alertHueSaturation(hueSat, light));
Mock::VerifyAndClearExpectations(handler.get());
}
// Colormode xy
{
const nlohmann::json state
= {{"colormode", "xy"}, {"on", true}, {"xy", {0.1, 0.1}}, {"sat", 100}, {"bri", 254}};
light.getState()["state"] = state;
EXPECT_CALL(Const(light), getColorXY())
.Times(AnyNumber())
.WillRepeatedly(Return(XYBrightness {{0.1f, 0.1f}, 1.f}));
TestTransaction reverseTransaction = light.transaction().setColor(XY {0.1f, 0.1f}).setTransition(1);
InSequence s;
EXPECT_CALL(light, setColorHueSaturation(hueSat, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(false));
EXPECT_FALSE(SimpleColorHueStrategy().alertHueSaturation(hueSat, light));
light.getState()["state"] = state;
Mock::VerifyAndClearExpectations(handler.get());
EXPECT_CALL(light, setColorHueSaturation(hueSat, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectSuccessfulPut(handler, Exactly(1));
EXPECT_TRUE(SimpleColorHueStrategy().alertHueSaturation(hueSat, light));
Mock::VerifyAndClearExpectations(handler.get());
}
// Colormode xy, off
{
const nlohmann::json state = {{"colormode", "xy"}, {"on", false}, {"xy", {0., 1.}}, {"sat", 100}, {"bri", 254}};
EXPECT_CALL(Const(light), getColorXY())
.Times(AnyNumber())
.WillRepeatedly(Return(XYBrightness {{0.f, 1.f}, 1.f}));
light.getState()["state"] = state;
TestTransaction reverseTransaction = light.transaction().setColor(XY {0.f, 1.f}).setOn(false).setTransition(1);
InSequence s;
EXPECT_CALL(light, setColorHueSaturation(hueSat, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectSuccessfulPut(handler, Exactly(1));
EXPECT_TRUE(SimpleColorHueStrategy().alertHueSaturation(hueSat, light));
Mock::VerifyAndClearExpectations(handler.get());
}
}
TEST(SimpleColorHueStrategy, alertXY)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight light(handler);
// Invalid colormode
{
light.getState()["state"]["colormode"] = "invalid";
light.getState()["state"]["on"] = false;
EXPECT_EQ(false, SimpleColorHueStrategy().alertXY({{0.1f, 0.1f}, 1.f}, light));
}
const XYBrightness xy {{0.1f, 0.1f}, 1.f};
// Needs to update the state so transactions are correctly trimmed
const auto setColorLambda = [&](const XYBrightness& xy, int transition) {
light.getState()["state"]["colormode"] = "xy";
light.getState()["state"]["on"] = true;
light.getState()["state"]["xy"] = {xy.xy.x, xy.xy.y};
light.getState()["state"]["bri"] = static_cast(std::round(xy.brightness * 254.f));
return true;
};
// Colormode hs
{
const nlohmann::json state
= {{"colormode", "hs"}, {"on", true}, {"xy", {0.1, 0.1}}, {"hue", 200}, {"sat", 100}, {"bri", 254}};
light.getState()["state"] = state;
EXPECT_CALL(Const(light), getBrightness()).Times(AnyNumber()).WillRepeatedly(Return(254));
HueSaturation hueSat {200, 100};
EXPECT_CALL(Const(light), getColorHueSaturation()).Times(AnyNumber()).WillRepeatedly(Return(hueSat));
TestTransaction reverseTransaction = light.transaction().setColor(hueSat).setTransition(1);
// Set color fails
{
EXPECT_CALL(light, setColorXY(xy, 1)).WillOnce(Return(false));
EXPECT_FALSE(SimpleColorHueStrategy().alertXY(xy, light));
}
// Alert call fails
{
InSequence s;
EXPECT_CALL(light, setColorXY(xy, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(false));
EXPECT_FALSE(SimpleColorHueStrategy().alertXY(xy, light));
}
light.getState()["state"] = state;
// Reverse transaction fails
{
InSequence s;
EXPECT_CALL(light, setColorXY(xy, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectPut(handler).WillOnce(Return(nlohmann::json::object()));
EXPECT_FALSE(SimpleColorHueStrategy().alertXY(xy, light));
}
light.getState()["state"] = state;
Mock::VerifyAndClearExpectations(handler.get());
// Successful
{
InSequence s;
EXPECT_CALL(light, setColorXY(xy, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectSuccessfulPut(handler, Exactly(1));
EXPECT_TRUE(SimpleColorHueStrategy().alertXY(xy, light));
}
Mock::VerifyAndClearExpectations(handler.get());
}
// Colormode hs, off
{
const nlohmann::json state
= {{"colormode", "hs"}, {"on", false}, {"xy", {0.1, 0.1}}, {"hue", 200}, {"sat", 100}, {"bri", 254}};
light.getState()["state"] = state;
TestTransaction reverseTransaction
= light.transaction().setColor(HueSaturation {200, 100}).setOn(false).setTransition(1);
InSequence s;
EXPECT_CALL(light, setColorXY(xy, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectSuccessfulPut(handler, Exactly(1));
EXPECT_TRUE(SimpleColorHueStrategy().alertXY(xy, light));
Mock::VerifyAndClearExpectations(handler.get());
}
// Colormode xy
{
const nlohmann::json state
= {{"colormode", "xy"}, {"on", true}, {"xy", {0.1, 0.1}}, {"sat", 100}, {"bri", 254}};
light.getState()["state"] = state;
EXPECT_CALL(Const(light), getColorXY()).Times(AnyNumber()).WillRepeatedly(Return(xy));
// No reverse transaction sent, because color already matches
EXPECT_CALL(light, setColorXY(xy, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(false));
EXPECT_FALSE(SimpleColorHueStrategy().alertXY(xy, light));
light.getState()["state"] = state;
Mock::VerifyAndClearExpectations(handler.get());
EXPECT_CALL(light, setColorXY(xy, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
EXPECT_TRUE(SimpleColorHueStrategy().alertXY(xy, light));
Mock::VerifyAndClearExpectations(handler.get());
}
// Colormode xy, off
{
const nlohmann::json state = {{"colormode", "xy"}, {"on", false}, {"xy", {0., 1.}}, {"sat", 100}, {"bri", 254}};
EXPECT_CALL(Const(light), getColorXY())
.Times(AnyNumber())
.WillRepeatedly(Return(XYBrightness {{0.f, 1.f}, 1.f}));
light.getState()["state"] = state;
// Brightness unchanged, so not requested
TestTransaction reverseTransaction = light.transaction().setColor(XY {0.f, 1.f}).setOn(false).setTransition(1);
InSequence s;
EXPECT_CALL(light, setColorXY(xy, 1)).WillOnce(Invoke(setColorLambda));
EXPECT_CALL(light, alert()).WillOnce(Return(true));
reverseTransaction.expectSuccessfulPut(handler, Exactly(1));
EXPECT_TRUE(SimpleColorHueStrategy().alertXY(xy, light));
Mock::VerifyAndClearExpectations(handler.get());
}
}
TEST(SimpleColorHueStrategy, getColorHueSaturation)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight test_light(handler);
test_light.getState()["state"]["hue"] = 5000;
test_light.getState()["state"]["sat"] = 128;
EXPECT_EQ((HueSaturation {5000, 128}), SimpleColorHueStrategy().getColorHueSaturation(test_light));
test_light.getState()["state"]["hue"] = 50000;
test_light.getState()["state"]["sat"] = 158;
EXPECT_EQ((HueSaturation {50000, 158}),
SimpleColorHueStrategy().getColorHueSaturation(static_cast(test_light)));
}
TEST(SimpleColorHueStrategy, getColorXY)
{
using namespace ::testing;
std::shared_ptr handler(std::make_shared());
EXPECT_CALL(
*handler, GETJson("/api/" + getBridgeUsername() + "/lights/1", nlohmann::json::object(), getBridgeIp(), 80))
.Times(AtLeast(1))
.WillRepeatedly(Return(nlohmann::json::object()));
MockLight test_light(handler);
test_light.getState()["state"]["xy"][0] = 0.1234;
test_light.getState()["state"]["xy"][1] = 0.1234;
test_light.getState()["state"]["bri"] = 254;
EXPECT_EQ((XYBrightness {{0.1234f, 0.1234f}, 1.f}), SimpleColorHueStrategy().getColorXY(test_light));
test_light.getState()["state"]["xy"][0] = 0.12;
test_light.getState()["state"]["xy"][1] = 0.6458;
EXPECT_EQ((XYBrightness {{0.12f, 0.6458f}, 1.f}), SimpleColorHueStrategy().getColorXY(Const(test_light)));
}