mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2023-10-10 13:36:59 +02:00
Philips Hue APIv2 support (#1637)
* Support Philips Hue APIv2 and refactoring * Fix MDNSBrower - if timeout during host resolvment occurs * Hue API v2 - Migrate database * Fix macOS build * Handle network timeout before any other error * Address CodeQL findings * Clean-up and Fixes * Only getProperties, if username is available * Option to layout by entertainment area center * Fix Wizard --------- Co-authored-by: Paulchen-Panther <16664240+Paulchen-Panther@users.noreply.github.com>
This commit is contained in:
parent
33722c9a09
commit
cd22d4454d
@ -86,6 +86,8 @@
|
|||||||
"conf_leds_layout_cl_bottomright": "Bottom Right (Corner)",
|
"conf_leds_layout_cl_bottomright": "Bottom Right (Corner)",
|
||||||
"conf_leds_layout_cl_cornergap": "Corner Gap",
|
"conf_leds_layout_cl_cornergap": "Corner Gap",
|
||||||
"conf_leds_layout_cl_edgegap": "Edge Gap",
|
"conf_leds_layout_cl_edgegap": "Edge Gap",
|
||||||
|
"conf_leds_layout_cl_entertainment": "Entertainment Area",
|
||||||
|
"conf_leds_layout_cl_entertainment_center": "Entertainment Area Center",
|
||||||
"conf_leds_layout_cl_gaglength": "Gap length",
|
"conf_leds_layout_cl_gaglength": "Gap length",
|
||||||
"conf_leds_layout_cl_gappos": "gap position",
|
"conf_leds_layout_cl_gappos": "gap position",
|
||||||
"conf_leds_layout_cl_hleddepth": "Horizontal LED depth",
|
"conf_leds_layout_cl_hleddepth": "Horizontal LED depth",
|
||||||
@ -618,7 +620,7 @@
|
|||||||
"edt_dev_spec_gpioBcm_title": "GPIO Pin",
|
"edt_dev_spec_gpioBcm_title": "GPIO Pin",
|
||||||
"edt_dev_spec_gpioMap_title": "GPIO mapping",
|
"edt_dev_spec_gpioMap_title": "GPIO mapping",
|
||||||
"edt_dev_spec_gpioNumber_title": "GPIO number",
|
"edt_dev_spec_gpioNumber_title": "GPIO number",
|
||||||
"edt_dev_spec_groupId_title": "Group ID",
|
"edt_dev_spec_groupId_title": "Group",
|
||||||
"edt_dev_spec_header_title": "Specific Settings",
|
"edt_dev_spec_header_title": "Specific Settings",
|
||||||
"edt_dev_spec_interpolation_title": "Interpolation",
|
"edt_dev_spec_interpolation_title": "Interpolation",
|
||||||
"edt_dev_spec_intervall_title": "Interval",
|
"edt_dev_spec_intervall_title": "Interval",
|
||||||
@ -683,6 +685,7 @@
|
|||||||
"edt_dev_spec_transistionTime_title": "Transition time",
|
"edt_dev_spec_transistionTime_title": "Transition time",
|
||||||
"edt_dev_spec_uid_title": "UID",
|
"edt_dev_spec_uid_title": "UID",
|
||||||
"edt_dev_spec_universe_title": "Universe",
|
"edt_dev_spec_universe_title": "Universe",
|
||||||
|
"edt_dev_spec_useAPIv2_title": "Use API v2",
|
||||||
"edt_dev_spec_useEntertainmentAPI_title": "Use Hue Entertainment API",
|
"edt_dev_spec_useEntertainmentAPI_title": "Use Hue Entertainment API",
|
||||||
"edt_dev_spec_useOrbSmoothing_title": "Use orb smoothing",
|
"edt_dev_spec_useOrbSmoothing_title": "Use orb smoothing",
|
||||||
"edt_dev_spec_useRgbwProtocol_title": "Use RGBW protocol",
|
"edt_dev_spec_useRgbwProtocol_title": "Use RGBW protocol",
|
||||||
@ -1087,7 +1090,7 @@
|
|||||||
"wiz_cololight_noprops": "Not able to get device properties - Define Hardware LED count manually",
|
"wiz_cololight_noprops": "Not able to get device properties - Define Hardware LED count manually",
|
||||||
"wiz_cololight_title": "Cololight Wizard",
|
"wiz_cololight_title": "Cololight Wizard",
|
||||||
"wiz_guideyou": "The $1 will guide you through the settings. Just press the button!",
|
"wiz_guideyou": "The $1 will guide you through the settings. Just press the button!",
|
||||||
"wiz_hue_blinkblue": "Let ID $1 light up blue",
|
"wiz_hue_blinkblue": "Let it light up",
|
||||||
"wiz_hue_clientkey": "Clientkey",
|
"wiz_hue_clientkey": "Clientkey",
|
||||||
"wiz_hue_create_user": "Create new User",
|
"wiz_hue_create_user": "Create new User",
|
||||||
"wiz_hue_desc1": "1. Hyperion searches automatically for a Hue-Bridge, in case it cannot find one you need to provide the hostname or IP-address and push the reload button. <br> 2. Provide a user ID, if you do not have one create a new one.",
|
"wiz_hue_desc1": "1. Hyperion searches automatically for a Hue-Bridge, in case it cannot find one you need to provide the hostname or IP-address and push the reload button. <br> 2. Provide a user ID, if you do not have one create a new one.",
|
||||||
|
@ -1021,11 +1021,6 @@ $(document).ready(function () {
|
|||||||
var generalOptions = window.serverSchema.properties.device;
|
var generalOptions = window.serverSchema.properties.device;
|
||||||
|
|
||||||
var ledType = $(this).val();
|
var ledType = $(this).val();
|
||||||
|
|
||||||
// philipshueentertainment backward fix
|
|
||||||
if (ledType == "philipshueentertainment")
|
|
||||||
ledType = "philipshue";
|
|
||||||
|
|
||||||
var specificOptions = window.serverSchema.properties.alldevices[ledType];
|
var specificOptions = window.serverSchema.properties.alldevices[ledType];
|
||||||
|
|
||||||
conf_editor = createJsonEditor('editor_container_leddevice', {
|
conf_editor = createJsonEditor('editor_container_leddevice', {
|
||||||
@ -1060,13 +1055,10 @@ $(document).ready(function () {
|
|||||||
$('#btn_led_device_wiz').off();
|
$('#btn_led_device_wiz').off();
|
||||||
|
|
||||||
if (ledType == "philipshue") {
|
if (ledType == "philipshue") {
|
||||||
$('#root_specificOptions_useEntertainmentAPI').on("change", function () {
|
var ledWizardType = ledType;
|
||||||
var ledWizardType = (this.checked) ? "philipshueentertainment" : ledType;
|
|
||||||
var data = { type: ledWizardType };
|
var data = { type: ledWizardType };
|
||||||
var hue_title = (this.checked) ? 'wiz_hue_e_title' : 'wiz_hue_title';
|
var hue_title = 'wiz_hue_title';
|
||||||
changeWizard(data, hue_title, startWizardPhilipsHue);
|
changeWizard(data, hue_title, startWizardPhilipsHue);
|
||||||
});
|
|
||||||
$("#root_specificOptions_useEntertainmentAPI").trigger("change");
|
|
||||||
}
|
}
|
||||||
else if (ledType == "atmoorb") {
|
else if (ledType == "atmoorb") {
|
||||||
var ledWizardType = (this.checked) ? "atmoorb" : ledType;
|
var ledWizardType = (this.checked) ? "atmoorb" : ledType;
|
||||||
|
@ -604,7 +604,7 @@ var lightPosTopLeft112 = { hmin: 0, hmax: 0.5, vmin: 0, vmax: 0.15 };
|
|||||||
var lightPosTopLeft121 = { hmin: 0.5, hmax: 1, vmin: 0, vmax: 0.15 };
|
var lightPosTopLeft121 = { hmin: 0.5, hmax: 1, vmin: 0, vmax: 0.15 };
|
||||||
var lightPosTopLeftNewMid = { hmin: 0.25, hmax: 0.75, vmin: 0, vmax: 0.15 };
|
var lightPosTopLeftNewMid = { hmin: 0.25, hmax: 0.75, vmin: 0, vmax: 0.15 };
|
||||||
|
|
||||||
function assignLightPos(id, pos, name) {
|
function assignLightPos(pos, name) {
|
||||||
var i = null;
|
var i = null;
|
||||||
|
|
||||||
if (pos === "top")
|
if (pos === "top")
|
||||||
@ -695,52 +695,50 @@ devicesProperties = {};
|
|||||||
|
|
||||||
var hueIPs = [];
|
var hueIPs = [];
|
||||||
var hueIPsinc = 0;
|
var hueIPsinc = 0;
|
||||||
var hueLights = null;
|
var hueLights = [];
|
||||||
var hueGroups = null;
|
var hueEntertainmentConfigs = [];
|
||||||
|
var hueEntertainmentServices = [];
|
||||||
var lightLocation = [];
|
var lightLocation = [];
|
||||||
var groupLights = [];
|
var groupLights = [];
|
||||||
|
var groupChannels = [];
|
||||||
var groupLightsLocations = [];
|
var groupLightsLocations = [];
|
||||||
var hueType = "philipshue";
|
var isAPIv2Ready = true;
|
||||||
|
var isEntertainmentReady = true;
|
||||||
|
|
||||||
function startWizardPhilipsHue(e) {
|
function startWizardPhilipsHue(e) {
|
||||||
if (typeof e.data.type != "undefined") hueType = e.data.type;
|
|
||||||
|
|
||||||
//create html
|
//create html
|
||||||
|
|
||||||
var hue_title = 'wiz_hue_title';
|
var hue_title = 'wiz_hue_title';
|
||||||
var hue_intro1 = 'wiz_hue_intro1';
|
var hue_intro1 = 'wiz_hue_e_intro1';
|
||||||
var hue_desc1 = 'wiz_hue_desc1';
|
var hue_desc1 = 'wiz_hue_desc1';
|
||||||
var hue_create_user = 'wiz_hue_create_user';
|
var hue_create_user = 'wiz_hue_create_user';
|
||||||
if (hueType == 'philipshueentertainment') {
|
|
||||||
hue_title = 'wiz_hue_e_title';
|
|
||||||
hue_intro1 = 'wiz_hue_e_intro1';
|
|
||||||
hue_desc1 = 'wiz_hue_e_desc1';
|
|
||||||
hue_create_user = 'wiz_hue_e_create_user';
|
|
||||||
}
|
|
||||||
$('#wiz_header').html('<i class="fa fa-magic fa-fw"></i>' + $.i18n(hue_title));
|
$('#wiz_header').html('<i class="fa fa-magic fa-fw"></i>' + $.i18n(hue_title));
|
||||||
$('#wizp1_body').html('<h4 style="font-weight:bold;text-transform:uppercase;">' + $.i18n(hue_title) + '</h4><p>' + $.i18n(hue_intro1) + '</p>');
|
$('#wizp1_body').html('<h4 style="font-weight:bold;text-transform:uppercase;">' + $.i18n(hue_title) + '</h4><p>' + $.i18n(hue_intro1) + '</p>');
|
||||||
$('#wizp1_footer').html('<button type="button" class="btn btn-primary" id="btn_wiz_cont"><i class="fa fa-fw fa-check"></i>' + $.i18n('general_btn_continue') + '</button><button type="button" class="btn btn-danger" data-dismiss="modal"><i class="fa fa-fw fa-close"></i>' + $.i18n('general_btn_cancel') + '</button>');
|
$('#wizp1_footer').html('<button type="button" class="btn btn-primary" id="btn_wiz_cont"><i class="fa fa-fw fa-check"></i>' + $.i18n('general_btn_continue') + '</button><button type="button" class="btn btn-danger" data-dismiss="modal"><i class="fa fa-fw fa-close"></i>' + $.i18n('general_btn_cancel') + '</button>');
|
||||||
$('#wizp2_body').html('<div id="wh_topcontainer"></div>');
|
$('#wizp2_body').html('<div id="wh_topcontainer"></div>');
|
||||||
|
|
||||||
var hidePort = "hidden-lg";
|
var topContainer_html = '<p class="text-left" style="font-weight:bold">' + $.i18n(hue_desc1) + '</p>' +
|
||||||
if (storedAccess === 'expert') {
|
|
||||||
hidePort = "";
|
|
||||||
}
|
|
||||||
|
|
||||||
$('#wh_topcontainer').append('<p class="text-left" style="font-weight:bold">' + $.i18n(hue_desc1) + '</p>' +
|
|
||||||
'<div class="row">' +
|
'<div class="row">' +
|
||||||
'<div class="col-md-2">' +
|
'<div class="col-md-2">' +
|
||||||
' <p class="text-left">' + $.i18n('wiz_hue_ip') + '</p></div>' +
|
' <p class="text-left">' + $.i18n('wiz_hue_ip') + '</p></div>' +
|
||||||
' <div class="col-md-7"><div class="input-group">' +
|
' <div class="col-md-7"><div class="input-group">' +
|
||||||
' <span class="input-group-addon" id="retry_bridge" style="cursor:pointer"><i class="fa fa-refresh"></i></span>' +
|
' <span class="input-group-addon" id="retry_bridge" style="cursor:pointer"><i class="fa fa-refresh"></i></span>' +
|
||||||
' <input type="text" class="input-group form-control" id="host" placeholder="' + $.i18n('wiz_hue_ip') + '"></div></div>' +
|
' <select id="hue_bridge_select" class="hue_bridge_sel_watch form-control">' + '</select>' + '</div></div>' +
|
||||||
' <div class="col-md-3 ' + hidePort + '"><div class="input-group">' +
|
' <div class="col-md-7"><div class="input-group">' +
|
||||||
' <span class="input-group-addon">:</span>' +
|
' <span class="input-group-addon"><i class="fa fa-arrow-right"></i></span>' +
|
||||||
' <input type="text" class="input-group form-control" id="port" placeholder="' + $.i18n('edt_conf_general_port_title') + '"></div></div>' +
|
' <input type="text" class="input-group form-control" id="host" placeholder="' + $.i18n('wiz_hue_ip') + '"></div></div>';
|
||||||
'</div><p><span style="font-weight:bold;color:red" id="wiz_hue_ipstate"></span><span style="font-weight:bold;" id="wiz_hue_discovered"></span></p>'
|
|
||||||
);
|
if (storedAccess === 'expert') {
|
||||||
$('#wh_topcontainer').append();
|
topContainer_html += '<div class="col-md-3"><div class="input-group">' +
|
||||||
$('#wh_topcontainer').append('<div class="form-group" id="usrcont" style="display:none"></div>');
|
'<span class="input-group-addon">:</span>' +
|
||||||
|
'<input type="text" class="input-group form-control" id="port" placeholder="' + $.i18n('edt_conf_general_port_title') + '"></div></div>';
|
||||||
|
}
|
||||||
|
|
||||||
|
topContainer_html += '</div><p><span style="font-weight:bold;color:red" id="wiz_hue_ipstate"></span><span style="font-weight:bold;" id="wiz_hue_discovered"></span></p>';
|
||||||
|
topContainer_html += '<div class="form-group" id="usrcont" style="display:none"></div>';
|
||||||
|
|
||||||
|
$('#wh_topcontainer').append(topContainer_html);
|
||||||
|
|
||||||
$('#usrcont').append('<div class="row"><div class="col-md-2"><p class="text-left">' + $.i18n('wiz_hue_username') + '</p ></div>' +
|
$('#usrcont').append('<div class="row"><div class="col-md-2"><p class="text-left">' + $.i18n('wiz_hue_username') + '</p ></div>' +
|
||||||
'<div class="col-md-7">' +
|
'<div class="col-md-7">' +
|
||||||
@ -751,23 +749,18 @@ function startWizardPhilipsHue(e) {
|
|||||||
'</div><input type="hidden" id="groupId">'
|
'</div><input type="hidden" id="groupId">'
|
||||||
);
|
);
|
||||||
|
|
||||||
if (hueType == 'philipshueentertainment') {
|
$('#usrcont').append('<div id="hue_client_key_r" class="row"><div class="col-md-2"><p class="text-left">' + $.i18n('wiz_hue_clientkey') +
|
||||||
$('#usrcont').append('<div class="row"><div class="col-md-2"><p class="text-left">' + $.i18n('wiz_hue_clientkey') +
|
|
||||||
'</p></div><div class="col-md-7"><input class="form-control" id="clientkey" type="text"></div></div><br>');
|
'</p></div><div class="col-md-7"><input class="form-control" id="clientkey" type="text"></div></div><br>');
|
||||||
}
|
|
||||||
|
|
||||||
$('#usrcont').append('<p><span style="font-weight:bold;color:red" id="wiz_hue_usrstate"></span><\p>' +
|
$('#usrcont').append('<p><span style="font-weight:bold;color:red" id="wiz_hue_usrstate"></span><\p>' +
|
||||||
'<button type="button" class="btn btn-primary" style="display:none" id="wiz_hue_create_user"> <i class="fa fa-fw fa-plus"></i>' + $.i18n(hue_create_user) + '</button>');
|
'<button type="button" class="btn btn-primary" style="display:none" id="wiz_hue_create_user"> <i class="fa fa-fw fa-plus"></i>' + $.i18n(hue_create_user) + '</button>');
|
||||||
|
|
||||||
if (hueType == 'philipshueentertainment') {
|
|
||||||
$('#wizp2_body').append('<div id="hue_grp_ids_t" style="display:none"><p class="text-left" style="font-weight:bold">' + $.i18n('wiz_hue_e_desc2') + '</p></div>');
|
$('#wizp2_body').append('<div id="hue_grp_ids_t" style="display:none"><p class="text-left" style="font-weight:bold">' + $.i18n('wiz_hue_e_desc2') + '</p></div>');
|
||||||
createTable("gidsh", "gidsb", "hue_grp_ids_t");
|
createTable("gidsh", "gidsb", "hue_grp_ids_t");
|
||||||
$('.gidsh').append(createTableRow([$.i18n('edt_dev_spec_groupId_title'), $.i18n('wiz_hue_e_use_group')], true));
|
$('.gidsh').append(createTableRow([$.i18n('edt_dev_spec_groupId_title'), ""], true));
|
||||||
|
|
||||||
$('#wizp2_body').append('<div id="hue_ids_t" style="display:none"><p class="text-left" style="font-weight:bold" id="hue_id_headline">' + $.i18n('wiz_hue_e_desc3') + '</p></div>');
|
$('#wizp2_body').append('<div id="hue_ids_t" style="display:none"><p class="text-left" style="font-weight:bold" id="hue_id_headline">' + $.i18n('wiz_hue_e_desc3') + '</p></div>');
|
||||||
}
|
|
||||||
else {
|
|
||||||
$('#wizp2_body').append('<div id="hue_ids_t" style="display:none"><p class="text-left" style="font-weight:bold" id="hue_id_headline">' + $.i18n('wiz_hue_desc2') + '</p></div>');
|
|
||||||
}
|
|
||||||
createTable("lidsh", "lidsb", "hue_ids_t");
|
createTable("lidsh", "lidsb", "hue_ids_t");
|
||||||
$('.lidsh').append(createTableRow([$.i18n('edt_dev_spec_lightid_title'), $.i18n('wiz_pos'), $.i18n('wiz_identify')], true));
|
$('.lidsh').append(createTableRow([$.i18n('edt_dev_spec_lightid_title'), $.i18n('wiz_pos'), $.i18n('wiz_identify')], true));
|
||||||
$('#wizp2_footer').html('<button type="button" class="btn btn-primary" id="btn_wiz_save" style="display:none"><i class="fa fa-fw fa-save"></i>' + $.i18n('general_btn_save') + '</button><button type="button" class="btn btn-danger" id="btn_wiz_abort"><i class="fa fa-fw fa-close"></i>' + $.i18n('general_btn_cancel') + '</button>');
|
$('#wizp2_footer').html('<button type="button" class="btn btn-primary" id="btn_wiz_save" style="display:none"><i class="fa fa-fw fa-save"></i>' + $.i18n('general_btn_save') + '</button><button type="button" class="btn btn-danger" id="btn_wiz_abort"><i class="fa fa-fw fa-close"></i>' + $.i18n('general_btn_cancel') + '</button>');
|
||||||
@ -793,14 +786,27 @@ function startWizardPhilipsHue(e) {
|
|||||||
|
|
||||||
function checkHueBridge(cb, hueUser) {
|
function checkHueBridge(cb, hueUser) {
|
||||||
var usr = (typeof hueUser != "undefined") ? hueUser : 'config';
|
var usr = (typeof hueUser != "undefined") ? hueUser : 'config';
|
||||||
if (usr == 'config') $('#wiz_hue_discovered').html("");
|
if (usr === 'config') {
|
||||||
|
$('#wiz_hue_discovered').html("");
|
||||||
|
}
|
||||||
|
|
||||||
if (hueIPs[hueIPsinc]) {
|
if (hueIPs[hueIPsinc]) {
|
||||||
var host = hueIPs[hueIPsinc].host;
|
var host = hueIPs[hueIPsinc].host;
|
||||||
var port = hueIPs[hueIPsinc].port;
|
var port = hueIPs[hueIPsinc].port;
|
||||||
|
|
||||||
|
if (usr != '')
|
||||||
|
{
|
||||||
getProperties_hue_bridge(cb, decodeURIComponent(host), port, usr);
|
getProperties_hue_bridge(cb, decodeURIComponent(host), port, usr);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cb(false, usr);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
$('#port').val(443);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function checkBridgeResult(reply, usr) {
|
function checkBridgeResult(reply, usr) {
|
||||||
@ -811,37 +817,51 @@ function checkBridgeResult(reply, usr) {
|
|||||||
$('#port').val(hueIPs[hueIPsinc].port)
|
$('#port').val(hueIPs[hueIPsinc].port)
|
||||||
|
|
||||||
$('#usrcont').toggle(true);
|
$('#usrcont').toggle(true);
|
||||||
checkHueBridge(checkUserResult, $('#user').val() ? $('#user').val() : "newdeveloper");
|
|
||||||
}
|
checkHueBridge(checkUserResult, $('#user').val());
|
||||||
else {
|
|
||||||
//increment and check again
|
|
||||||
if (hueIPs.length - 1 > hueIPsinc) {
|
|
||||||
hueIPsinc++;
|
|
||||||
checkHueBridge(checkBridgeResult);
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
$('#usrcont').toggle(false);
|
$('#usrcont').toggle(false);
|
||||||
$('#wiz_hue_ipstate').html($.i18n('wiz_hue_failure_ip'));
|
$('#wiz_hue_ipstate').html($.i18n('wiz_hue_failure_ip'));
|
||||||
}
|
}
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
function checkUserResult(reply, usr) {
|
function checkUserResult(reply, username) {
|
||||||
$('#usrcont').toggle(true);
|
$('#usrcont').toggle(true);
|
||||||
if (reply) {
|
|
||||||
$('#user').val(usr);
|
|
||||||
if (hueType == 'philipshueentertainment' && $('#clientkey').val() == "") {
|
|
||||||
|
|
||||||
|
var hue_create_user = 'wiz_hue_e_create_user';
|
||||||
|
if (!isEntertainmentReady) {
|
||||||
|
hue_create_user = 'wiz_hue_create_user';
|
||||||
|
$('#hue_client_key_r').toggle(false);
|
||||||
|
} else {
|
||||||
|
$('#hue_client_key_r').toggle(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
$('#wiz_hue_create_user').text($.i18n(hue_create_user));
|
||||||
|
$('#wiz_hue_create_user').toggle(true);
|
||||||
|
|
||||||
|
if (reply) {
|
||||||
|
$('#user').val(username);
|
||||||
|
|
||||||
|
if (isEntertainmentReady && $('#clientkey').val() == "") {
|
||||||
$('#wiz_hue_usrstate').html($.i18n('wiz_hue_e_clientkey_needed'));
|
$('#wiz_hue_usrstate').html($.i18n('wiz_hue_e_clientkey_needed'));
|
||||||
$('#wiz_hue_create_user').toggle(true);
|
$('#wiz_hue_create_user').toggle(true);
|
||||||
} else {
|
} else {
|
||||||
$('#wiz_hue_usrstate').html("");
|
$('#wiz_hue_usrstate').html("");
|
||||||
$('#wiz_hue_create_user').toggle(false);
|
$('#wiz_hue_create_user').toggle(false);
|
||||||
if (hueType == 'philipshue') {
|
|
||||||
get_hue_lights();
|
if (isEntertainmentReady) {
|
||||||
}
|
$('#hue_id_headline').text($.i18n('wiz_hue_e_desc3'));
|
||||||
if (hueType == 'philipshueentertainment') {
|
$('#hue_grp_ids_t').toggle(true);
|
||||||
get_hue_groups();
|
|
||||||
|
get_hue_groups(username);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
$('#hue_id_headline').text($.i18n('wiz_hue_desc2'));
|
||||||
|
$('#hue_grp_ids_t').toggle(false);
|
||||||
|
|
||||||
|
get_hue_lights(username);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -852,22 +872,73 @@ function checkUserResult(reply, usr) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
function useGroupId(id) {
|
function useGroupId(id, username) {
|
||||||
$('#groupId').val(id);
|
$('#groupId').val(hueEntertainmentConfigs[id].id);
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
var group = hueEntertainmentConfigs[id];
|
||||||
|
|
||||||
|
groupLights = [];
|
||||||
|
for (const light of group.light_services) {
|
||||||
|
groupLights.push(light.rid);
|
||||||
|
}
|
||||||
|
|
||||||
|
groupChannels = [];
|
||||||
|
for (const channel of group.channels) {
|
||||||
|
groupChannels.push(channel);
|
||||||
|
}
|
||||||
|
|
||||||
|
groupLightsLocations = [];
|
||||||
|
for (const location of group.locations.service_locations) {
|
||||||
|
groupLightsLocations.push(location);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
//Ensure ligthIDs are strings
|
//Ensure ligthIDs are strings
|
||||||
groupLights = hueGroups[id].lights.map(num => {
|
groupLights = hueEntertainmentConfigs[id].lights.map(num => {
|
||||||
return String(num);
|
return String(num);
|
||||||
});
|
});
|
||||||
|
|
||||||
groupLightsLocations = hueGroups[id].locations;
|
var lightLocations = hueEntertainmentConfigs[id].locations;
|
||||||
get_hue_lights();
|
for (var locationID in lightLocations) {
|
||||||
|
var lightLocation = {};
|
||||||
|
|
||||||
|
let position = {
|
||||||
|
x: lightLocations[locationID][0],
|
||||||
|
y: lightLocations[locationID][1],
|
||||||
|
z: lightLocations[locationID][2]
|
||||||
|
};
|
||||||
|
lightLocation.position = position;
|
||||||
|
|
||||||
|
groupLightsLocations.push(lightLocation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get_hue_lights(username);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function updateBridgeDetails(properties) {
|
||||||
|
var ledDeviceProperties = properties.config;
|
||||||
|
|
||||||
|
if (!jQuery.isEmptyObject(ledDeviceProperties)) {
|
||||||
|
isEntertainmentReady = properties.isEntertainmentReady;
|
||||||
|
isAPIv2Ready = properties.isAPIv2Ready;
|
||||||
|
|
||||||
|
if (ledDeviceProperties.name && ledDeviceProperties.bridgeid && ledDeviceProperties.modelid) {
|
||||||
|
$('#wiz_hue_discovered').html(
|
||||||
|
"Bridge: " + ledDeviceProperties.name +
|
||||||
|
", Modelid: " + ledDeviceProperties.modelid +
|
||||||
|
", Firmware: " + ledDeviceProperties.swversion + "<br/>" +
|
||||||
|
"API-Version: " + ledDeviceProperties.apiversion +
|
||||||
|
", Entertainment: " + (isEntertainmentReady ? "✓" : "-") +
|
||||||
|
", APIv2: " + (isAPIv2Ready ? "✓" : "-")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async function discover_hue_bridges() {
|
async function discover_hue_bridges() {
|
||||||
$('#wiz_hue_ipstate').html($.i18n('edt_dev_spec_devices_discovery_inprogress'));
|
$('#wiz_hue_ipstate').html($.i18n('edt_dev_spec_devices_discovery_inprogress'));
|
||||||
$('#wiz_hue_discovered').html("")
|
|
||||||
|
// $('#wiz_hue_discovered').html("")
|
||||||
const res = await requestLedDeviceDiscovery('philipshue');
|
const res = await requestLedDeviceDiscovery('philipshue');
|
||||||
if (res && !res.error) {
|
if (res && !res.error) {
|
||||||
const r = res.info;
|
const r = res.info;
|
||||||
@ -903,11 +974,6 @@ async function discover_hue_bridges() {
|
|||||||
port = device.port;
|
port = device.port;
|
||||||
}
|
}
|
||||||
|
|
||||||
//Remap https port to http port until Hue-API v2 is supported
|
|
||||||
if (port == 443) {
|
|
||||||
port = 80;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (host) {
|
if (host) {
|
||||||
|
|
||||||
if (!hueIPs.some(item => item.host === host)) {
|
if (!hueIPs.some(item => item.host === host)) {
|
||||||
@ -916,22 +982,39 @@ async function discover_hue_bridges() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$('#wiz_hue_ipstate').html("");
|
$('#wiz_hue_ipstate').html("");
|
||||||
$('#host').val(hueIPs[hueIPsinc].host)
|
$('#host').val(hueIPs[hueIPsinc].host)
|
||||||
$('#port').val(hueIPs[hueIPsinc].port)
|
$('#port').val(hueIPs[hueIPsinc].port)
|
||||||
|
|
||||||
|
$('#hue_bridge_select').html("");
|
||||||
|
|
||||||
|
for (var key in hueIPs) {
|
||||||
|
$('#hue_bridge_select').append(createSelOpt(key, hueIPs[key].host));
|
||||||
|
}
|
||||||
|
|
||||||
|
$('.hue_bridge_sel_watch').on("click", function () {
|
||||||
|
hueIPsinc = $(this).val();
|
||||||
|
|
||||||
|
var name = $("#hue_bridge_select option:selected").text();
|
||||||
|
$('#host').val(name);
|
||||||
|
$('#port').val(hueIPs[hueIPsinc].port)
|
||||||
|
|
||||||
var usr = $('#user').val();
|
var usr = $('#user').val();
|
||||||
if (usr != "") {
|
if (usr != "") {
|
||||||
checkHueBridge(checkUserResult, usr);
|
checkHueBridge(checkUserResult, usr);
|
||||||
} else {
|
} else {
|
||||||
checkHueBridge(checkBridgeResult);
|
checkHueBridge(checkBridgeResult);
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$('.hue_bridge_sel_watch').click();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function getProperties_hue_bridge(cb, hostAddress, port, username, resourceFilter) {
|
async function getProperties_hue_bridge(cb, hostAddress, port, username, resourceFilter) {
|
||||||
let params = { host: hostAddress, user: username, filter: resourceFilter };
|
let params = { host: hostAddress, username: username, filter: resourceFilter };
|
||||||
if (port !== 'undefined') {
|
if (port !== 'undefined') {
|
||||||
params.port = parseInt(port);
|
params.port = parseInt(port);
|
||||||
}
|
}
|
||||||
@ -945,23 +1028,27 @@ async function getProperties_hue_bridge(cb, hostAddress, port, username, resourc
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Use device's properties, if properties in chache
|
// Use device's properties, if properties in chache
|
||||||
if (devicesProperties[ledType][key]) {
|
if (devicesProperties[ledType][key] && devicesProperties[ledType][key][username]) {
|
||||||
|
updateBridgeDetails(devicesProperties[ledType][key]);
|
||||||
cb(true, username);
|
cb(true, username);
|
||||||
} else {
|
} else {
|
||||||
const res = await requestLedDeviceProperties(ledType, params);
|
const res = await requestLedDeviceProperties(ledType, params);
|
||||||
|
|
||||||
|
|
||||||
if (res && !res.error) {
|
if (res && !res.error) {
|
||||||
var ledDeviceProperties = res.info.properties;
|
var ledDeviceProperties = res.info.properties;
|
||||||
if (!jQuery.isEmptyObject(ledDeviceProperties)) {
|
if (!jQuery.isEmptyObject(ledDeviceProperties)) {
|
||||||
|
|
||||||
|
devicesProperties[ledType][key] = {};
|
||||||
|
devicesProperties[ledType][key][username] = ledDeviceProperties;
|
||||||
|
|
||||||
|
isAPIv2Ready = res.info.isAPIv2Ready;
|
||||||
|
devicesProperties[ledType][key].isAPIv2Ready = isAPIv2Ready;
|
||||||
|
isEntertainmentReady = res.info.isEntertainmentReady;
|
||||||
|
devicesProperties[ledType][key].isEntertainmentReady = isEntertainmentReady;
|
||||||
|
|
||||||
|
updateBridgeDetails(devicesProperties[ledType][key]);
|
||||||
if (username === "config") {
|
if (username === "config") {
|
||||||
if (ledDeviceProperties.name && ledDeviceProperties.bridgeid && ledDeviceProperties.modelid) {
|
|
||||||
$('#wiz_hue_discovered').html("Bridge: " + ledDeviceProperties.name + ", Modelid: " + ledDeviceProperties.modelid + ", API-Version: " + ledDeviceProperties.apiversion);
|
|
||||||
cb(true);
|
cb(true);
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
devicesProperties[ledType][key] = ledDeviceProperties;
|
|
||||||
cb(true, username);
|
cb(true, username);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -973,12 +1060,12 @@ async function getProperties_hue_bridge(cb, hostAddress, port, username, resourc
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function identify_hue_device(hostAddress, port, username, id) {
|
async function identify_hue_device(hostAddress, port, username, name, id, id_v1) {
|
||||||
var disabled = $('#btn_wiz_save').is(':disabled');
|
var disabled = $('#btn_wiz_save').is(':disabled');
|
||||||
// Take care that new record cannot be save during background process
|
// Take care that new record cannot be save during background process
|
||||||
$('#btn_wiz_save').prop('disabled', true);
|
$('#btn_wiz_save').prop('disabled', true);
|
||||||
|
|
||||||
let params = { host: decodeURIComponent(hostAddress), user: username, lightId: id };
|
let params = { host: decodeURIComponent(hostAddress), username: username, lightName: decodeURIComponent(name), lightId: id, lightId_v1: id_v1 };
|
||||||
|
|
||||||
if (port !== 'undefined') {
|
if (port !== 'undefined') {
|
||||||
params.port = parseInt(port);
|
params.port = parseInt(port);
|
||||||
@ -1003,12 +1090,10 @@ function beginWizardHue() {
|
|||||||
$('#user').val(usr);
|
$('#user').val(usr);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hueType == 'philipshueentertainment') {
|
|
||||||
var clkey = eV("clientkey");
|
var clkey = eV("clientkey");
|
||||||
if (clkey != "") {
|
if (clkey != "") {
|
||||||
$('#clientkey').val(clkey);
|
$('#clientkey').val(clkey);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
//check if host is empty/reachable/search for bridge
|
//check if host is empty/reachable/search for bridge
|
||||||
if (eV("host") == "") {
|
if (eV("host") == "") {
|
||||||
@ -1022,13 +1107,13 @@ function beginWizardHue() {
|
|||||||
$('#host').val(host);
|
$('#host').val(host);
|
||||||
|
|
||||||
var port = eV("port");
|
var port = eV("port");
|
||||||
if (port == 0) {
|
if (port > 0) {
|
||||||
$('#port').val(80);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
$('#port').val(port);
|
$('#port').val(port);
|
||||||
}
|
}
|
||||||
hueIPs.unshift({ host: host, port: port });
|
else {
|
||||||
|
$('#port').val('');
|
||||||
|
}
|
||||||
|
hueIPs.push({ host: host, port: port });
|
||||||
|
|
||||||
if (usr != "") {
|
if (usr != "") {
|
||||||
checkHueBridge(checkUserResult, usr);
|
checkHueBridge(checkUserResult, usr);
|
||||||
@ -1038,18 +1123,18 @@ function beginWizardHue() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
$('#retry_bridge').off().on('click', function () {
|
$('#retry_bridge').off().on('click', function () {
|
||||||
|
var host = $('#host').val();
|
||||||
|
var port = parseInt($('#port').val());
|
||||||
|
|
||||||
if ($('#host').val() != "") {
|
if (host != "") {
|
||||||
|
|
||||||
hueIPs = [];
|
var idx = hueIPs.findIndex(item => item.host === host && item.port === port);
|
||||||
hueIPsinc = 0;
|
if (idx === -1) {
|
||||||
|
hueIPs.push({ host: host, port: port });
|
||||||
var port = $('#port').val();
|
hueIPsinc = hueIPs.length - 1;
|
||||||
if (isNaN(port) || port < 1 || port > 65535) {
|
} else {
|
||||||
port = 80;
|
hueIPsinc = idx;
|
||||||
$('#port').val(80);
|
|
||||||
}
|
}
|
||||||
hueIPs.push({ host: $('#host').val(), port: port });
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
discover_hue_bridges();
|
discover_hue_bridges();
|
||||||
@ -1064,29 +1149,177 @@ function beginWizardHue() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
$('#retry_usr').off().on('click', function () {
|
$('#retry_usr').off().on('click', function () {
|
||||||
checkHueBridge(checkUserResult, $('#user').val() ? $('#user').val() : "newdeveloper");
|
checkHueBridge(checkUserResult, $('#user').val());
|
||||||
});
|
});
|
||||||
|
|
||||||
$('#wiz_hue_create_user').off().on('click', function () {
|
$('#wiz_hue_create_user').off().on('click', function () {
|
||||||
if ($('#host').val() != "") {
|
|
||||||
hueIPs.unshift({ host: $('#host').val(), port: $('#port').val() });
|
|
||||||
}
|
|
||||||
createHueUser();
|
createHueUser();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function assignLightEntertainmentPos(isFocusCenter, position, name, id) {
|
||||||
|
|
||||||
|
var x = position.x;
|
||||||
|
var z = position.z;
|
||||||
|
|
||||||
|
if (isFocusCenter) {
|
||||||
|
// Map lights as in centered range -0.5 to 0.5
|
||||||
|
if (x < -0.5) {
|
||||||
|
x = -0.5;
|
||||||
|
} else if (x > 0.5) {
|
||||||
|
x = 0.5;
|
||||||
|
}
|
||||||
|
if (z < -0.5) {
|
||||||
|
z = -0.5;
|
||||||
|
} else if (z > 0.5) {
|
||||||
|
z = 0.5;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Map lights as in full range -1 to 1
|
||||||
|
x /= 2;
|
||||||
|
z /= 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
var h = x + 0.5;
|
||||||
|
var v = -z + 0.5;
|
||||||
|
|
||||||
|
var hmin = h - 0.05;
|
||||||
|
var hmax = h + 0.05;
|
||||||
|
var vmin = v - 0.05;
|
||||||
|
var vmax = v + 0.05;
|
||||||
|
|
||||||
|
let layoutObject = {
|
||||||
|
hmin: hmin < 0 ? 0 : hmin,
|
||||||
|
hmax: hmax > 1 ? 1 : hmax,
|
||||||
|
vmin: vmin < 0 ? 0 : vmin,
|
||||||
|
vmax: vmax > 1 ? 1 : vmax,
|
||||||
|
name: name
|
||||||
|
};
|
||||||
|
|
||||||
|
if (id) {
|
||||||
|
layoutObject.name += "_" + id;
|
||||||
|
}
|
||||||
|
return layoutObject;
|
||||||
|
}
|
||||||
|
|
||||||
|
function assignSegmentedLightPos(segment, position, name) {
|
||||||
|
var layoutObjects = [];
|
||||||
|
|
||||||
|
var segTotalLength = 0;
|
||||||
|
for (var key in segment) {
|
||||||
|
|
||||||
|
segTotalLength += segment[key].length;
|
||||||
|
}
|
||||||
|
|
||||||
|
var min;
|
||||||
|
var max;
|
||||||
|
var horizontal = true;
|
||||||
|
|
||||||
|
var layoutObject = assignLightPos(position, name);
|
||||||
|
if (position === "left" || position === "right") {
|
||||||
|
// vertical distribution
|
||||||
|
min = layoutObject.vmin;
|
||||||
|
max = layoutObject.vmax;
|
||||||
|
horizontal = false;
|
||||||
|
|
||||||
|
} else {
|
||||||
|
// horizontal distribution
|
||||||
|
min = layoutObject.hmin;
|
||||||
|
max = layoutObject.hmax;
|
||||||
|
}
|
||||||
|
|
||||||
|
var step = (max - min) / segTotalLength;
|
||||||
|
var start = min;
|
||||||
|
|
||||||
|
for (var key in segment) {
|
||||||
|
min = start;
|
||||||
|
max = round(start + segment[key].length * step);
|
||||||
|
|
||||||
|
if (horizontal) {
|
||||||
|
layoutObject.hmin = min;
|
||||||
|
layoutObject.hmax = max;
|
||||||
|
} else {
|
||||||
|
layoutObject.vmin = min;
|
||||||
|
layoutObject.vmax = max;
|
||||||
|
}
|
||||||
|
layoutObject.name = name + "_" + key;
|
||||||
|
layoutObjects.push(JSON.parse(JSON.stringify(layoutObject)));
|
||||||
|
|
||||||
|
start = max;
|
||||||
|
}
|
||||||
|
|
||||||
|
return layoutObjects;
|
||||||
|
}
|
||||||
|
|
||||||
$('#btn_wiz_save').off().on("click", function () {
|
$('#btn_wiz_save').off().on("click", function () {
|
||||||
var hueLedConfig = [];
|
var hueLedConfig = [];
|
||||||
var finalLightIds = [];
|
var finalLightIds = [];
|
||||||
|
var channelNumber = 0;
|
||||||
|
|
||||||
//create hue led config
|
//create hue led config
|
||||||
for (var key in hueLights) {
|
for (var key in groupLights) {
|
||||||
if (hueType == 'philipshueentertainment') {
|
var lightId = groupLights[key];
|
||||||
if (groupLights.indexOf(key) == -1) continue;
|
|
||||||
|
if ($('#hue_' + lightId).val() != "disabled") {
|
||||||
|
finalLightIds.push(lightId);
|
||||||
|
|
||||||
|
var lightName;
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
var light = hueLights.find(light => light.id === lightId);
|
||||||
|
lightName = light.metadata.name;
|
||||||
|
} else {
|
||||||
|
lightName = hueLights[lightId].name;
|
||||||
|
}
|
||||||
|
|
||||||
|
var position = $('#hue_' + lightId).val();
|
||||||
|
var lightIdx = groupLights.indexOf(lightId);
|
||||||
|
var lightLocation = groupLightsLocations[lightIdx];
|
||||||
|
|
||||||
|
var serviceID;
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
serviceID = lightLocation.service.rid;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (position.startsWith("entertainment")) {
|
||||||
|
|
||||||
|
// Layout per entertainment area definition at bridge
|
||||||
|
var isFocusCenter = false;
|
||||||
|
if (position === "entertainment_center") {
|
||||||
|
isFocusCenter = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
|
||||||
|
groupChannels.forEach((channel) => {
|
||||||
|
if (channel.members[0].service.rid === serviceID) {
|
||||||
|
var layoutObject = assignLightEntertainmentPos(isFocusCenter, channel.position, lightName, channel.channel_id);
|
||||||
|
hueLedConfig.push(JSON.parse(JSON.stringify(layoutObject)));
|
||||||
|
++channelNumber;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
var layoutObject = assignLightEntertainmentPos(isFocusCenter, lightLocation.position, lightName);
|
||||||
|
hueLedConfig.push(JSON.parse(JSON.stringify(layoutObject)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// Layout per manual settings
|
||||||
|
var maxSegments = 1;
|
||||||
|
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
var service = hueEntertainmentServices.find(service => service.id === serviceID);
|
||||||
|
maxSegments = service.segments.max_segments;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (maxSegments > 1) {
|
||||||
|
var segment = service.segments.segments;
|
||||||
|
var layoutObjects = assignSegmentedLightPos(segment, position, lightName);
|
||||||
|
hueLedConfig.push(...layoutObjects);
|
||||||
|
} else {
|
||||||
|
var layoutObject = assignLightPos(position, lightName);
|
||||||
|
hueLedConfig.push(JSON.parse(JSON.stringify(layoutObject)));
|
||||||
|
}
|
||||||
|
channelNumber += maxSegments;
|
||||||
}
|
}
|
||||||
if ($('#hue_' + key).val() != "disabled") {
|
|
||||||
finalLightIds.push(key);
|
|
||||||
var idx_content = assignLightPos(key, $('#hue_' + key).val(), hueLights[key].name);
|
|
||||||
hueLedConfig.push(JSON.parse(JSON.stringify(idx_content)));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1121,7 +1354,7 @@ function beginWizardHue() {
|
|||||||
d.brightnessFactor = parseFloat(eV("brightnessFactor", 1));
|
d.brightnessFactor = parseFloat(eV("brightnessFactor", 1));
|
||||||
|
|
||||||
d.clientkey = $('#clientkey').val();
|
d.clientkey = $('#clientkey').val();
|
||||||
d.groupId = parseInt($('#groupId').val());
|
d.groupId = $('#groupId').val();
|
||||||
d.blackLightsTimeout = parseInt(eV("blackLightsTimeout", 5000));
|
d.blackLightsTimeout = parseInt(eV("blackLightsTimeout", 5000));
|
||||||
d.brightnessMin = parseFloat(eV("brightnessMin", 0));
|
d.brightnessMin = parseFloat(eV("brightnessMin", 0));
|
||||||
d.brightnessMax = parseFloat(eV("brightnessMax", 1));
|
d.brightnessMax = parseFloat(eV("brightnessMax", 1));
|
||||||
@ -1134,8 +1367,16 @@ function beginWizardHue() {
|
|||||||
d.enableAttempts = parseInt(conf_editor.getEditor("root.generalOptions.enableAttempts").getValue());
|
d.enableAttempts = parseInt(conf_editor.getEditor("root.generalOptions.enableAttempts").getValue());
|
||||||
d.enableAttemptsInterval = parseInt(conf_editor.getEditor("root.generalOptions.enableAttemptsInterval").getValue());
|
d.enableAttemptsInterval = parseInt(conf_editor.getEditor("root.generalOptions.enableAttemptsInterval").getValue());
|
||||||
|
|
||||||
if (hueType == 'philipshue') {
|
d.useEntertainmentAPI = isEntertainmentReady;
|
||||||
d.useEntertainmentAPI = false;
|
d.useAPIv2 = isAPIv2Ready;
|
||||||
|
|
||||||
|
if (isEntertainmentReady) {
|
||||||
|
d.hardwareLedCount = channelNumber;
|
||||||
|
if (window.serverConfig.device.type !== d.type) {
|
||||||
|
//smoothing on, if new device
|
||||||
|
sc.smoothing = { enable: true };
|
||||||
|
}
|
||||||
|
} else {
|
||||||
d.hardwareLedCount = finalLightIds.length;
|
d.hardwareLedCount = finalLightIds.length;
|
||||||
d.verbose = false;
|
d.verbose = false;
|
||||||
if (window.serverConfig.device.type !== d.type) {
|
if (window.serverConfig.device.type !== d.type) {
|
||||||
@ -1144,15 +1385,6 @@ function beginWizardHue() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hueType == 'philipshueentertainment') {
|
|
||||||
d.useEntertainmentAPI = true;
|
|
||||||
d.hardwareLedCount = groupLights.length;
|
|
||||||
if (window.serverConfig.device.type !== d.type) {
|
|
||||||
//smoothing on, if new device
|
|
||||||
sc.smoothing = { enable: true };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
window.serverConfig.device = d;
|
window.serverConfig.device = d;
|
||||||
|
|
||||||
requestWriteConfig(sc, true);
|
requestWriteConfig(sc, true);
|
||||||
@ -1163,7 +1395,6 @@ function beginWizardHue() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function createHueUser() {
|
function createHueUser() {
|
||||||
|
|
||||||
var host = hueIPs[hueIPsinc].host;
|
var host = hueIPs[hueIPsinc].host;
|
||||||
var port = hueIPs[hueIPsinc].port;
|
var port = hueIPs[hueIPsinc].port;
|
||||||
|
|
||||||
@ -1208,7 +1439,8 @@ function createHueUser() {
|
|||||||
conf_editor.getEditor("root.specificOptions.host").setValue(host);
|
conf_editor.getEditor("root.specificOptions.host").setValue(host);
|
||||||
conf_editor.getEditor("root.specificOptions.port").setValue(port);
|
conf_editor.getEditor("root.specificOptions.port").setValue(port);
|
||||||
}
|
}
|
||||||
if (hueType == 'philipshueentertainment') {
|
|
||||||
|
if (isEntertainmentReady) {
|
||||||
var clientkey = response.clientkey;
|
var clientkey = response.clientkey;
|
||||||
if (clientkey != 'undefined') {
|
if (clientkey != 'undefined') {
|
||||||
$('#clientkey').val(clientkey);
|
$('#clientkey').val(clientkey);
|
||||||
@ -1230,37 +1462,52 @@ function createHueUser() {
|
|||||||
}, retryInterval * 1000);
|
}, retryInterval * 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_hue_groups() {
|
function get_hue_groups(username) {
|
||||||
|
|
||||||
var host = hueIPs[hueIPsinc].host;
|
var host = hueIPs[hueIPsinc].host;
|
||||||
|
|
||||||
if (devicesProperties['philipshue'][host]) {
|
if (devicesProperties['philipshue'][host] && devicesProperties['philipshue'][host][username]) {
|
||||||
var ledProperties = devicesProperties['philipshue'][host];
|
var ledProperties = devicesProperties['philipshue'][host][username];
|
||||||
|
|
||||||
if (!jQuery.isEmptyObject(ledProperties)) {
|
if (isAPIv2Ready) {
|
||||||
hueGroups = ledProperties.groups;
|
if (!jQuery.isEmptyObject(ledProperties.data)) {
|
||||||
if (Object.keys(hueGroups).length > 0) {
|
if (Object.keys(ledProperties.data).length > 0) {
|
||||||
|
hueEntertainmentConfigs = ledProperties.data.filter(config => {
|
||||||
|
return config.type === "entertainment_configuration";
|
||||||
|
});
|
||||||
|
hueEntertainmentServices = ledProperties.data.filter(config => {
|
||||||
|
return (config.type === "entertainment" && config.renderer === true);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (!jQuery.isEmptyObject(ledProperties.groups)) {
|
||||||
|
hueEntertainmentConfigs = [];
|
||||||
|
var hueGroups = ledProperties.groups;
|
||||||
|
for (var groupid in hueGroups) {
|
||||||
|
if (hueGroups[groupid].type == 'Entertainment') {
|
||||||
|
hueGroups[groupid].id = groupid;
|
||||||
|
hueEntertainmentConfigs.push(hueGroups[groupid]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Object.keys(hueEntertainmentConfigs).length > 0) {
|
||||||
|
|
||||||
$('.lidsb').html("");
|
$('.lidsb').html("");
|
||||||
$('#wh_topcontainer').toggle(false);
|
$('#wh_topcontainer').toggle(false);
|
||||||
$('#hue_grp_ids_t').toggle(true);
|
$('#hue_grp_ids_t').toggle(true);
|
||||||
|
|
||||||
var gC = 0;
|
for (var groupid in hueEntertainmentConfigs) {
|
||||||
for (var groupid in hueGroups) {
|
$('.gidsb').append(createTableRow([groupid + ' (' + hueEntertainmentConfigs[groupid].name + ')', '<button class="btn btn-sm btn-primary" onClick=useGroupId("' + groupid + '","' + username + '")>' + $.i18n('wiz_hue_e_use_group') + '</button>']));
|
||||||
if (hueGroups[groupid].type == 'Entertainment') {
|
|
||||||
$('.gidsb').append(createTableRow([groupid + ' (' + hueGroups[groupid].name + ')', '<button class="btn btn-sm btn-primary" onClick=useGroupId(' + groupid + ')>' + $.i18n('wiz_hue_e_use_groupid', groupid) + '</button>']));
|
|
||||||
gC++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (gC == 0) {
|
|
||||||
noAPISupport('wiz_hue_e_noegrpids');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
noAPISupport('wiz_hue_e_noegrpids', username);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function noAPISupport(txt) {
|
function noAPISupport(txt, username) {
|
||||||
showNotification('danger', $.i18n('wiz_hue_e_title'), $.i18n('wiz_hue_e_noapisupport_hint'));
|
showNotification('danger', $.i18n('wiz_hue_e_title'), $.i18n('wiz_hue_e_noapisupport_hint'));
|
||||||
conf_editor.getEditor("root.specificOptions.useEntertainmentAPI").setValue(false);
|
conf_editor.getEditor("root.specificOptions.useEntertainmentAPI").setValue(false);
|
||||||
$("#root_specificOptions_useEntertainmentAPI").trigger("change");
|
$("#root_specificOptions_useEntertainmentAPI").trigger("change");
|
||||||
@ -1269,21 +1516,32 @@ function noAPISupport(txt) {
|
|||||||
var txt = (txt) ? $.i18n(txt) : $.i18n('wiz_hue_e_nogrpids');
|
var txt = (txt) ? $.i18n(txt) : $.i18n('wiz_hue_e_nogrpids');
|
||||||
$('<p style="font-weight:bold;color:red;">' + txt + '<br />' + $.i18n('wiz_hue_e_noapisupport') + '</p>').insertBefore('#wizp2_body #hue_ids_t');
|
$('<p style="font-weight:bold;color:red;">' + txt + '<br />' + $.i18n('wiz_hue_e_noapisupport') + '</p>').insertBefore('#wizp2_body #hue_ids_t');
|
||||||
$('#hue_id_headline').html($.i18n('wiz_hue_desc2'));
|
$('#hue_id_headline').html($.i18n('wiz_hue_desc2'));
|
||||||
hueType = 'philipshue';
|
|
||||||
get_hue_lights();
|
get_hue_lights(username);
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_hue_lights() {
|
function get_hue_lights(username) {
|
||||||
|
|
||||||
var host = hueIPs[hueIPsinc].host;
|
var host = hueIPs[hueIPsinc].host;
|
||||||
|
|
||||||
if (devicesProperties['philipshue'][host]) {
|
if (devicesProperties['philipshue'][host] && devicesProperties['philipshue'][host][username]) {
|
||||||
var ledProperties = devicesProperties['philipshue'][host];
|
var ledProperties = devicesProperties['philipshue'][host][username];
|
||||||
|
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
if (!jQuery.isEmptyObject(ledProperties.data)) {
|
||||||
|
if (Object.keys(ledProperties.data).length > 0) {
|
||||||
|
hueLights = ledProperties.data.filter(config => {
|
||||||
|
return config.type === "light";
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
if (!jQuery.isEmptyObject(ledProperties.lights)) {
|
if (!jQuery.isEmptyObject(ledProperties.lights)) {
|
||||||
hueLights = ledProperties.lights;
|
hueLights = ledProperties.lights;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (Object.keys(hueLights).length > 0) {
|
if (Object.keys(hueLights).length > 0) {
|
||||||
if (hueType == 'philipshue') {
|
if (!isEntertainmentReady) {
|
||||||
$('#wh_topcontainer').toggle(false);
|
$('#wh_topcontainer').toggle(false);
|
||||||
}
|
}
|
||||||
$('#hue_ids_t, #btn_wiz_save').toggle(true);
|
$('#hue_ids_t, #btn_wiz_save').toggle(true);
|
||||||
@ -1299,21 +1557,41 @@ function get_hue_lights() {
|
|||||||
"lightPosBottomLeft112", "lightPosBottomLeftNewMid", "lightPosBottomLeft121"
|
"lightPosBottomLeft112", "lightPosBottomLeftNewMid", "lightPosBottomLeft121"
|
||||||
];
|
];
|
||||||
|
|
||||||
if (hueType == 'philipshue') {
|
if (isEntertainmentReady) {
|
||||||
|
lightOptions.unshift("entertainment_center");
|
||||||
|
lightOptions.unshift("entertainment");
|
||||||
|
} else {
|
||||||
lightOptions.unshift("disabled");
|
lightOptions.unshift("disabled");
|
||||||
|
groupLights = Object.keys(hueLights);
|
||||||
}
|
}
|
||||||
|
|
||||||
$('.lidsb').html("");
|
$('.lidsb').html("");
|
||||||
var pos = "";
|
|
||||||
for (var lightid in hueLights) {
|
|
||||||
if (hueType == 'philipshueentertainment') {
|
|
||||||
if (groupLights.indexOf(lightid) == -1) continue;
|
|
||||||
|
|
||||||
if (groupLightsLocations.hasOwnProperty(lightid)) {
|
var pos = "";
|
||||||
lightLocation = groupLightsLocations[lightid];
|
for (var id in groupLights) {
|
||||||
var x = lightLocation[0];
|
var lightId = groupLights[id];
|
||||||
var y = lightLocation[1];
|
var lightId_v1 = "/lights/" + lightId;
|
||||||
var z = lightLocation[2];
|
|
||||||
|
var lightName;
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
var light = hueLights.find(light => light.id === lightId);
|
||||||
|
lightName = light.metadata.name;
|
||||||
|
lightId_v1 = light.id_v1;
|
||||||
|
} else {
|
||||||
|
lightName = hueLights[lightId].name;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isEntertainmentReady) {
|
||||||
|
var lightLocation = {};
|
||||||
|
lightLocation = groupLightsLocations[id];
|
||||||
|
if (lightLocation) {
|
||||||
|
if (isAPIv2Ready) {
|
||||||
|
pos = 0;
|
||||||
|
} else {
|
||||||
|
var x = lightLocation.position.x;
|
||||||
|
var y = lightLocation.position.y;
|
||||||
|
var z = lightLocation.position.z;
|
||||||
|
|
||||||
var xval = (x < 0) ? "left" : "right";
|
var xval = (x < 0) ? "left" : "right";
|
||||||
if (z != 1 && x >= -0.25 && x <= 0.25) xval = "";
|
if (z != 1 && x >= -0.25 && x <= 0.25) xval = "";
|
||||||
switch (z) {
|
switch (z) {
|
||||||
@ -1329,6 +1607,8 @@ function get_hue_lights() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var options = "";
|
var options = "";
|
||||||
for (var opt in lightOptions) {
|
for (var opt in lightOptions) {
|
||||||
var val = lightOptions[opt];
|
var val = lightOptions[opt];
|
||||||
@ -1337,12 +1617,13 @@ function get_hue_lights() {
|
|||||||
if (pos == val) options += ' selected="selected"';
|
if (pos == val) options += ' selected="selected"';
|
||||||
options += '>' + $.i18n(txt + val) + '</option>';
|
options += '>' + $.i18n(txt + val) + '</option>';
|
||||||
}
|
}
|
||||||
$('.lidsb').append(createTableRow([lightid + ' (' + hueLights[lightid].name + ')', '<select id="hue_' + lightid + '" class="hue_sel_watch form-control">'
|
|
||||||
|
$('.lidsb').append(createTableRow([id + ' (' + lightName + ')', '<select id="hue_' + lightId + '" class="hue_sel_watch form-control">'
|
||||||
+ options
|
+ options
|
||||||
+ '</select>', '<button class="btn btn-sm btn-primary" onClick=identify_hue_device("' + encodeURIComponent($("#host").val()) + '","' + $('#port').val() + '","' + $("#user").val() + '",' + lightid + ')>' + $.i18n('wiz_hue_blinkblue', lightid) + '</button>']));
|
+ '</select>', '<button class="btn btn-sm btn-primary" onClick=identify_hue_device("' + encodeURIComponent($("#host").val()) + '","' + $('#port').val() + '","' + $("#user").val() + '","' + encodeURIComponent(lightName) + '","' + lightId + '","' + lightId_v1 + '")>' + $.i18n('wiz_hue_blinkblue', id) + '</button>']));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hueType != 'philipshueentertainment') {
|
if (!isEntertainmentReady) {
|
||||||
$('.hue_sel_watch').on("change", function () {
|
$('.hue_sel_watch').on("change", function () {
|
||||||
var cC = 0;
|
var cC = 0;
|
||||||
for (var key in hueLights) {
|
for (var key in hueLights) {
|
||||||
@ -1361,7 +1642,6 @@ function get_hue_lights() {
|
|||||||
$('#wizp2_body').append(txt);
|
$('#wizp2_body').append(txt);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function abortConnection(UserInterval) {
|
function abortConnection(UserInterval) {
|
||||||
@ -1437,7 +1717,7 @@ function beginWizardYeelight() {
|
|||||||
|
|
||||||
finalLights.push(lights[key]);
|
finalLights.push(lights[key]);
|
||||||
|
|
||||||
var idx_content = assignLightPos(key, $('#yee_' + key).val(), name);
|
var idx_content = assignLightPos($('#yee_' + key).val(), name);
|
||||||
yeelightLedConfig.push(JSON.parse(JSON.stringify(idx_content)));
|
yeelightLedConfig.push(JSON.parse(JSON.stringify(idx_content)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1733,7 +2013,7 @@ function beginWizardAtmoOrb() {
|
|||||||
if (lights[key].host !== "")
|
if (lights[key].host !== "")
|
||||||
name += ':' + lights[key].host;
|
name += ':' + lights[key].host;
|
||||||
|
|
||||||
var idx_content = assignLightPos(key, $('#orb_' + key).val(), name);
|
var idx_content = assignLightPos($('#orb_' + key).val(), name);
|
||||||
atmoorbLedConfig.push(JSON.parse(JSON.stringify(idx_content)));
|
atmoorbLedConfig.push(JSON.parse(JSON.stringify(idx_content)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -723,6 +723,7 @@ bool SettingsManager::handleConfigUpgrade(QJsonObject& config)
|
|||||||
}
|
}
|
||||||
|
|
||||||
//Migration steps for versions <= 2.0.13
|
//Migration steps for versions <= 2.0.13
|
||||||
|
_previousVersion = targetVersion;
|
||||||
targetVersion.setVersion("2.0.13");
|
targetVersion.setVersion("2.0.13");
|
||||||
if (_previousVersion <= targetVersion)
|
if (_previousVersion <= targetVersion)
|
||||||
{
|
{
|
||||||
@ -774,6 +775,60 @@ bool SettingsManager::handleConfigUpgrade(QJsonObject& config)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Migration steps for versions <= 2.0.16
|
||||||
|
_previousVersion = targetVersion;
|
||||||
|
targetVersion.setVersion("2.0.16");
|
||||||
|
if (_previousVersion <= targetVersion)
|
||||||
|
{
|
||||||
|
Info(_log, "Instance [%u]: Migrate from version [%s] to version [%s] or later", _instance, _previousVersion.getVersion().c_str(), targetVersion.getVersion().c_str());
|
||||||
|
|
||||||
|
// Have Hostname/IP-address separate from port for LED-Devices
|
||||||
|
if (config.contains("device"))
|
||||||
|
{
|
||||||
|
QJsonObject newDeviceConfig = config["device"].toObject();
|
||||||
|
|
||||||
|
if (newDeviceConfig.contains("type"))
|
||||||
|
{
|
||||||
|
QString type = newDeviceConfig["type"].toString();
|
||||||
|
if ( type == "philipshue")
|
||||||
|
{
|
||||||
|
if (newDeviceConfig.contains("groupId"))
|
||||||
|
{
|
||||||
|
if (newDeviceConfig["groupId"].isDouble())
|
||||||
|
{
|
||||||
|
int groupID = newDeviceConfig["groupId"].toInt();
|
||||||
|
newDeviceConfig["groupId"] = QString::number(groupID);
|
||||||
|
migrated = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newDeviceConfig.contains("lightIds"))
|
||||||
|
{
|
||||||
|
QJsonArray lightIds = newDeviceConfig.value( "lightIds").toArray();
|
||||||
|
// Iterate through the JSON array and update integer values to strings
|
||||||
|
for (int i = 0; i < lightIds.size(); ++i) {
|
||||||
|
QJsonValue value = lightIds.at(i);
|
||||||
|
if (value.isDouble())
|
||||||
|
{
|
||||||
|
int lightId = value.toInt();
|
||||||
|
lightIds.replace(i, QString::number(lightId));
|
||||||
|
migrated = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
newDeviceConfig["lightIds"] = lightIds;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (migrated)
|
||||||
|
{
|
||||||
|
config["device"] = newDeviceConfig;
|
||||||
|
Debug(_log, "LED-Device records migrated");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return migrated;
|
return migrated;
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -16,31 +16,6 @@
|
|||||||
#include "ProviderRestApi.h"
|
#include "ProviderRestApi.h"
|
||||||
#include "ProviderUdpSSL.h"
|
#include "ProviderUdpSSL.h"
|
||||||
|
|
||||||
//Streaming message header and payload definition
|
|
||||||
const uint8_t HEADER[] =
|
|
||||||
{
|
|
||||||
'H', 'u', 'e', 'S', 't', 'r', 'e', 'a', 'm', //protocol
|
|
||||||
0x01, 0x00, //version 1.0
|
|
||||||
0x01, //sequence number 1
|
|
||||||
0x00, 0x00, //Reserved write 0’s
|
|
||||||
0x01, //xy Brightness
|
|
||||||
0x00, // Reserved, write 0’s
|
|
||||||
};
|
|
||||||
|
|
||||||
const uint8_t PAYLOAD_PER_LIGHT[] =
|
|
||||||
{
|
|
||||||
0x01, 0x00, 0x06, //light ID
|
|
||||||
//color: 16 bpc
|
|
||||||
0xff, 0xff,
|
|
||||||
0xff, 0xff,
|
|
||||||
0xff, 0xff,
|
|
||||||
/*
|
|
||||||
(message.R >> 8) & 0xff, message.R & 0xff,
|
|
||||||
(message.G >> 8) & 0xff, message.G & 0xff,
|
|
||||||
(message.B >> 8) & 0xff, message.B & 0xff
|
|
||||||
*/
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A XY color point in the color space of the hue system without brightness.
|
* A XY color point in the color space of the hue system without brightness.
|
||||||
*/
|
*/
|
||||||
@ -145,13 +120,19 @@ public:
|
|||||||
/// Constructs the light.
|
/// Constructs the light.
|
||||||
///
|
///
|
||||||
/// @param log the logger
|
/// @param log the logger
|
||||||
/// @param bridge the bridge
|
/// @param useApiV2 make use of Hue API version 2
|
||||||
/// @param id the light id
|
/// @param id the light id
|
||||||
|
/// @param lightAttributes the light's attributes as provied by the Hue Bridge
|
||||||
|
/// @param onBlackTimeToPowerOff Timeframe of Black output that triggers powering off the light
|
||||||
|
/// @param onBlackTimeToPowerOn Timeframe of non Black output that triggers powering on the light
|
||||||
///
|
///
|
||||||
PhilipsHueLight(Logger* log, int id, QJsonObject values, int ledidx,
|
PhilipsHueLight(Logger* log, bool useApiV2, const QString& id, const QJsonObject& lightAttributes,
|
||||||
int onBlackTimeToPowerOff,
|
int onBlackTimeToPowerOff,
|
||||||
int onBlackTimeToPowerOn);
|
int onBlackTimeToPowerOn);
|
||||||
|
|
||||||
|
void setDeviceDetails(const QJsonObject& details);
|
||||||
|
void setEntertainmentSrvDetails(const QJsonObject& details);
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @param on
|
/// @param on
|
||||||
///
|
///
|
||||||
@ -167,7 +148,14 @@ public:
|
|||||||
///
|
///
|
||||||
void setColor(const CiColor& color);
|
void setColor(const CiColor& color);
|
||||||
|
|
||||||
int getId() const;
|
QString getId() const;
|
||||||
|
QString getdeviceId() const;
|
||||||
|
QString getProduct() const;
|
||||||
|
QString getModel() const;
|
||||||
|
QString getName() const;
|
||||||
|
QString getArcheType() const;
|
||||||
|
|
||||||
|
int getMaxSegments() const;
|
||||||
|
|
||||||
bool getOnOffState() const;
|
bool getOnOffState() const;
|
||||||
int getTransitionTime() const;
|
int getTransitionTime() const;
|
||||||
@ -179,7 +167,7 @@ public:
|
|||||||
CiColorTriangle getColorSpace() const;
|
CiColorTriangle getColorSpace() const;
|
||||||
|
|
||||||
void saveOriginalState(const QJsonObject& values);
|
void saveOriginalState(const QJsonObject& values);
|
||||||
QString getOriginalState() const;
|
QJsonObject getOriginalState() const;
|
||||||
|
|
||||||
bool isBusy();
|
bool isBusy();
|
||||||
bool isBlack(bool isBlack);
|
bool isBlack(bool isBlack);
|
||||||
@ -189,24 +177,30 @@ public:
|
|||||||
private:
|
private:
|
||||||
|
|
||||||
Logger* _log;
|
Logger* _log;
|
||||||
/// light id
|
bool _useApiV2;
|
||||||
int _id;
|
|
||||||
int _ledidx;
|
QString _id;
|
||||||
|
QString _deviceId;
|
||||||
|
QString _product;
|
||||||
|
QString _model;
|
||||||
|
QString _name;
|
||||||
|
QString _archeType;
|
||||||
|
QString _gamutType;
|
||||||
|
|
||||||
|
int _maxSegments;
|
||||||
|
|
||||||
bool _on;
|
bool _on;
|
||||||
int _transitionTime;
|
int _transitionTime;
|
||||||
CiColor _color;
|
CiColor _color;
|
||||||
bool _hasColor;
|
bool _hasColor;
|
||||||
/// darkes blue color in hue lamp GAMUT = black
|
/// darkes blue color in hue lamp GAMUT = black
|
||||||
CiColor _colorBlack;
|
CiColor _colorBlack;
|
||||||
/// The model id of the hue lamp which is used to determine the color space.
|
|
||||||
QString _modelId;
|
|
||||||
QString _lightname;
|
|
||||||
CiColorTriangle _colorSpace;
|
CiColorTriangle _colorSpace;
|
||||||
|
|
||||||
/// The json string of the original state.
|
/// The json string of the original state.
|
||||||
QJsonObject _originalStateJSON;
|
QJsonObject _originalStateJSON;
|
||||||
|
|
||||||
QString _originalState;
|
QJsonObject _originalState;
|
||||||
CiColor _originalColor;
|
CiColor _originalColor;
|
||||||
qint64 _lastSendColorTime;
|
qint64 _lastSendColorTime;
|
||||||
qint64 _lastBlackTime;
|
qint64 _lastBlackTime;
|
||||||
@ -242,23 +236,40 @@ public:
|
|||||||
QJsonDocument get(const QString& route);
|
QJsonDocument get(const QString& route);
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Perform a REST-API POST
|
/// @brief Perform a REST-API GET
|
||||||
///
|
///
|
||||||
/// @param route the route of the POST request.
|
/// @param routeElements the route's elements of the GET request.
|
||||||
/// @param content the content of the POST request.
|
|
||||||
///
|
///
|
||||||
QJsonDocument put(const QString& route, const QString& content, bool supressError = false);
|
/// @return the content of the GET request.
|
||||||
|
///
|
||||||
|
QJsonDocument get(const QStringList& routeElements);
|
||||||
|
|
||||||
QJsonDocument getLightState( int lightId);
|
///
|
||||||
void setLightState( int lightId = 0, const QString &state = "");
|
/// @brief Perform a REST-API PUT
|
||||||
|
///
|
||||||
|
/// @param routeElements the route's elements of the PUT request.
|
||||||
|
/// @param content the content of the PUT request.
|
||||||
|
/// @param supressError Treat an error as a warning
|
||||||
|
///
|
||||||
|
/// @return the content of the PUT request.
|
||||||
|
///
|
||||||
|
QJsonDocument put(const QStringList& routeElements, const QJsonObject& content, bool supressError = false);
|
||||||
|
|
||||||
QMap<int,QJsonObject> getLightMap() const;
|
QJsonDocument retrieveBridgeDetails();
|
||||||
|
QJsonObject getDeviceDetails(const QString& deviceId);
|
||||||
|
QJsonObject getEntertainmentSrvDetails(const QString& deviceId);
|
||||||
|
|
||||||
QMap<int,QJsonObject> getGroupMap() const;
|
QJsonObject getLightDetails(const QString& lightId);
|
||||||
|
QJsonDocument setLightState(const QString& lightId, const QJsonObject& state);
|
||||||
|
|
||||||
QString getGroupName(int groupId = 0) const;
|
QMap<QString,QJsonObject> getDevicesMap() const;
|
||||||
|
QMap<QString,QJsonObject> getLightMap() const;
|
||||||
|
QMap<QString,QJsonObject> getGroupMap() const;
|
||||||
|
QMap<QString,QJsonObject> getEntertainmentMap() const;
|
||||||
|
|
||||||
QJsonArray getGroupLights(int groupId = 0) const;
|
QString getGroupName(const QString& groupId) const;
|
||||||
|
QStringList getGroupLights(const QString& groupId) const;
|
||||||
|
int getGroupChannelsCount(const QString& groupId) const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
@ -338,23 +349,41 @@ protected:
|
|||||||
///
|
///
|
||||||
QJsonObject addAuthorization(const QJsonObject& params) override;
|
QJsonObject addAuthorization(const QJsonObject& params) override;
|
||||||
|
|
||||||
|
bool isApiEntertainmentReady(const QString& apiVersion);
|
||||||
|
bool isAPIv2Ready (int swVersion);
|
||||||
|
|
||||||
|
int getFirmwareVerion() { return _deviceFirmwareVersion; }
|
||||||
|
void setBridgeDetails( const QJsonDocument &doc, bool isLogging = false );
|
||||||
|
|
||||||
|
void setBaseApiEnvironment(bool apiV2 = true, const QString& path = "");
|
||||||
|
|
||||||
|
QJsonDocument getGroupDetails( const QString& groupId );
|
||||||
|
QJsonDocument setGroupState( const QString& groupId, bool state);
|
||||||
|
|
||||||
|
bool isStreamOwner(const QString &streamOwner) const;
|
||||||
|
|
||||||
|
bool initDevicesMap();
|
||||||
|
bool initLightsMap();
|
||||||
|
bool initGroupsMap();
|
||||||
|
bool initEntertainmentSrvsMap();
|
||||||
|
|
||||||
|
void log(const char* msg, const char* type, ...) const;
|
||||||
|
|
||||||
|
bool configureSsl();
|
||||||
|
const int * getCiphersuites() const override;
|
||||||
|
|
||||||
///REST-API wrapper
|
///REST-API wrapper
|
||||||
ProviderRestApi* _restApi;
|
ProviderRestApi* _restApi;
|
||||||
int _apiPort;
|
int _apiPort;
|
||||||
/// User name for the API ("newdeveloper")
|
/// User name for the API ("newdeveloper")
|
||||||
QString _authToken;
|
QString _authToken;
|
||||||
|
QString _applicationID;
|
||||||
|
|
||||||
bool _useHueEntertainmentAPI;
|
bool _useEntertainmentAPI;
|
||||||
|
bool _useApiV2;
|
||||||
|
bool _isAPIv2Ready;
|
||||||
|
|
||||||
QJsonDocument getGroupState( int groupId );
|
bool _isDiyHue;
|
||||||
QJsonDocument setGroupState( int groupId, bool state);
|
|
||||||
|
|
||||||
bool isStreamOwner(const QString &streamOwner) const;
|
|
||||||
bool initMaps();
|
|
||||||
|
|
||||||
void log(const char* msg, const char* type, ...) const;
|
|
||||||
|
|
||||||
const int * getCiphersuites() const override;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
@ -364,16 +393,25 @@ private:
|
|||||||
///
|
///
|
||||||
/// @return A JSON structure holding a list of devices found
|
/// @return A JSON structure holding a list of devices found
|
||||||
///
|
///
|
||||||
QJsonArray discover();
|
QJsonArray discoverSsdp();
|
||||||
|
|
||||||
QJsonDocument getAllBridgeInfos();
|
QJsonDocument retrieveDeviceDetails(const QString& deviceId = "");
|
||||||
void setBridgeConfig( const QJsonDocument &doc );
|
QJsonDocument retrieveLightDetails(const QString& lightId = "");
|
||||||
|
QJsonDocument retrieveGroupDetails(const QString& groupId = "");
|
||||||
|
QJsonDocument retrieveEntertainmentSrvDetails(const QString& deviceId = "");
|
||||||
|
|
||||||
|
bool retrieveApplicationId();
|
||||||
|
|
||||||
|
void setDevicesMap( const QJsonDocument &doc );
|
||||||
void setLightsMap( const QJsonDocument &doc );
|
void setLightsMap( const QJsonDocument &doc );
|
||||||
void setGroupMap( const QJsonDocument &doc );
|
void setGroupMap( const QJsonDocument &doc );
|
||||||
|
void setEntertainmentSrvMap( const QJsonDocument &doc );
|
||||||
|
|
||||||
//Philips Hue Bridge details
|
//Philips Hue Bridge details
|
||||||
|
QString _deviceName;
|
||||||
|
QString _deviceBridgeId;
|
||||||
QString _deviceModel;
|
QString _deviceModel;
|
||||||
QString _deviceFirmwareVersion;
|
int _deviceFirmwareVersion;
|
||||||
QString _deviceAPIVersion;
|
QString _deviceAPIVersion;
|
||||||
|
|
||||||
uint _api_major;
|
uint _api_major;
|
||||||
@ -382,8 +420,12 @@ private:
|
|||||||
|
|
||||||
bool _isHueEntertainmentReady;
|
bool _isHueEntertainmentReady;
|
||||||
|
|
||||||
QMap<int,QJsonObject> _lightsMap;
|
QMap<QString,QJsonObject> _devicesMap;
|
||||||
QMap<int,QJsonObject> _groupsMap;
|
QMap<QString,QJsonObject> _lightsMap;
|
||||||
|
QMap<QString,QJsonObject> _groupsMap;
|
||||||
|
QMap<QString,QJsonObject> _entertainmentMap;
|
||||||
|
|
||||||
|
int _lightsCount;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -440,7 +482,7 @@ public:
|
|||||||
///
|
///
|
||||||
/// @return Number of device's LEDs
|
/// @return Number of device's LEDs
|
||||||
///
|
///
|
||||||
unsigned int getLightsCount() const { return _lightsCount; }
|
int getLightsCount() const { return _lightsCount; }
|
||||||
|
|
||||||
void setOnOffState(PhilipsHueLight& light, bool on, bool force = false);
|
void setOnOffState(PhilipsHueLight& light, bool on, bool force = false);
|
||||||
void setTransitionTime(PhilipsHueLight& light);
|
void setTransitionTime(PhilipsHueLight& light);
|
||||||
@ -547,18 +589,18 @@ private:
|
|||||||
|
|
||||||
bool setLights();
|
bool setLights();
|
||||||
|
|
||||||
/// creates new PhilipsHueLight(s) based on user lightid with bridge feedback
|
/// creates new PhilipsHueLight(s) based on user lightId with bridge feedback
|
||||||
///
|
///
|
||||||
/// @param map Map of lightid/value pairs of bridge
|
/// @param map Map of lightId/value pairs of bridge
|
||||||
///
|
///
|
||||||
bool updateLights(const QMap<int, QJsonObject> &map);
|
bool updateLights(const QMap<QString, QJsonObject> &map);
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Set the number of LEDs supported by the device.
|
/// @brief Set the number of LEDs supported by the device.
|
||||||
///
|
///
|
||||||
/// @rparam[in] Number of device's LEDs
|
/// @rparam[in] Number of device's LEDs
|
||||||
//
|
//
|
||||||
void setLightsCount( unsigned int lightsCount);
|
void setLightsCount(int lightsCount);
|
||||||
|
|
||||||
bool openStream();
|
bool openStream();
|
||||||
bool getStreamGroupState();
|
bool getStreamGroupState();
|
||||||
@ -566,10 +608,8 @@ private:
|
|||||||
bool startStream();
|
bool startStream();
|
||||||
bool stopStream();
|
bool stopStream();
|
||||||
|
|
||||||
void writeStream(bool flush = false);
|
|
||||||
int writeSingleLights(const std::vector<ColorRgb>& ledValues);
|
int writeSingleLights(const std::vector<ColorRgb>& ledValues);
|
||||||
|
int writeStreamData(const std::vector<ColorRgb>& ledValues, bool flush = false);
|
||||||
QByteArray prepareStreamData() const;
|
|
||||||
|
|
||||||
///
|
///
|
||||||
bool _switchOffOnBlack;
|
bool _switchOffOnBlack;
|
||||||
@ -582,12 +622,15 @@ private:
|
|||||||
bool _isInitLeds;
|
bool _isInitLeds;
|
||||||
|
|
||||||
/// Array of the light ids.
|
/// Array of the light ids.
|
||||||
std::vector<int> _lightIds;
|
QStringList _lightIds;
|
||||||
/// Array to save the lamps.
|
/// Array to save the lamps.
|
||||||
std::vector<PhilipsHueLight> _lights;
|
std::vector<PhilipsHueLight> _lights;
|
||||||
|
|
||||||
int _lightsCount;
|
int _lightsCount;
|
||||||
int _groupId;
|
int _channelsCount;
|
||||||
|
QString _groupId;
|
||||||
|
QString _groupName;
|
||||||
|
QString _streamOwner;
|
||||||
|
|
||||||
int _blackLightsTimeout;
|
int _blackLightsTimeout;
|
||||||
double _blackLevel;
|
double _blackLevel;
|
||||||
@ -595,15 +638,5 @@ private:
|
|||||||
int _onBlackTimeToPowerOn;
|
int _onBlackTimeToPowerOn;
|
||||||
bool _candyGamma;
|
bool _candyGamma;
|
||||||
|
|
||||||
// TODO: Check what is the correct class
|
|
||||||
uint32_t _handshake_timeout_min;
|
|
||||||
uint32_t _handshake_timeout_max;
|
|
||||||
bool _stopConnection;
|
|
||||||
|
|
||||||
QString _groupName;
|
|
||||||
QString _streamOwner;
|
|
||||||
|
|
||||||
qint64 _lastConfirm;
|
|
||||||
int _lastId;
|
|
||||||
bool _groupStreamState;
|
bool _groupStreamState;
|
||||||
};
|
};
|
||||||
|
@ -2,11 +2,18 @@
|
|||||||
#include "ProviderRestApi.h"
|
#include "ProviderRestApi.h"
|
||||||
|
|
||||||
// Qt includes
|
// Qt includes
|
||||||
|
#include <QObject>
|
||||||
#include <QEventLoop>
|
#include <QEventLoop>
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
|
|
||||||
|
#include <QList>
|
||||||
|
#include <QHash>
|
||||||
|
#include <QFile>
|
||||||
|
|
||||||
|
#include <QSslSocket>
|
||||||
|
|
||||||
//std includes
|
//std includes
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
@ -30,12 +37,12 @@ ProviderRestApi::ProviderRestApi(const QString& scheme, const QString& host, int
|
|||||||
: _log(Logger::getInstance("LEDDEVICE"))
|
: _log(Logger::getInstance("LEDDEVICE"))
|
||||||
, _networkManager(nullptr)
|
, _networkManager(nullptr)
|
||||||
, _requestTimeout(DEFAULT_REST_TIMEOUT)
|
, _requestTimeout(DEFAULT_REST_TIMEOUT)
|
||||||
|
,_isSeflSignedCertificateAccpeted(false)
|
||||||
{
|
{
|
||||||
_networkManager = new QNetworkAccessManager();
|
_networkManager = new QNetworkAccessManager();
|
||||||
#if (QT_VERSION >= QT_VERSION_CHECK(5, 9, 0))
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 9, 0))
|
||||||
_networkManager->setRedirectPolicy(QNetworkRequest::NoLessSafeRedirectPolicy);
|
_networkManager->setRedirectPolicy(QNetworkRequest::NoLessSafeRedirectPolicy);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
_apiUrl.setScheme(scheme);
|
_apiUrl.setScheme(scheme);
|
||||||
_apiUrl.setHost(host);
|
_apiUrl.setHost(host);
|
||||||
_apiUrl.setPort(port);
|
_apiUrl.setPort(port);
|
||||||
@ -46,7 +53,7 @@ ProviderRestApi::ProviderRestApi(const QString& scheme, const QString& host, int
|
|||||||
: ProviderRestApi(scheme, host, port, "") {}
|
: ProviderRestApi(scheme, host, port, "") {}
|
||||||
|
|
||||||
ProviderRestApi::ProviderRestApi(const QString& host, int port, const QString& basePath)
|
ProviderRestApi::ProviderRestApi(const QString& host, int port, const QString& basePath)
|
||||||
: ProviderRestApi("http", host, port, basePath) {}
|
: ProviderRestApi((port == 443) ? "https" : "http", host, port, basePath) {}
|
||||||
|
|
||||||
ProviderRestApi::ProviderRestApi(const QString& host, int port)
|
ProviderRestApi::ProviderRestApi(const QString& host, int port)
|
||||||
: ProviderRestApi(host, port, "") {}
|
: ProviderRestApi(host, port, "") {}
|
||||||
@ -59,18 +66,33 @@ ProviderRestApi::~ProviderRestApi()
|
|||||||
delete _networkManager;
|
delete _networkManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ProviderRestApi::setScheme(const QString& scheme)
|
||||||
|
{
|
||||||
|
_apiUrl.setScheme(scheme);
|
||||||
|
}
|
||||||
|
|
||||||
void ProviderRestApi::setUrl(const QUrl& url)
|
void ProviderRestApi::setUrl(const QUrl& url)
|
||||||
{
|
{
|
||||||
_apiUrl = url;
|
_apiUrl = url;
|
||||||
_basePath = url.path();
|
_basePath = url.path();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ProviderRestApi::setBasePath(const QStringList& pathElements)
|
||||||
|
{
|
||||||
|
setBasePath(pathElements.join(ONE_SLASH));
|
||||||
|
}
|
||||||
|
|
||||||
void ProviderRestApi::setBasePath(const QString& basePath)
|
void ProviderRestApi::setBasePath(const QString& basePath)
|
||||||
{
|
{
|
||||||
_basePath.clear();
|
_basePath.clear();
|
||||||
appendPath(_basePath, basePath);
|
appendPath(_basePath, basePath);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ProviderRestApi::clearBasePath()
|
||||||
|
{
|
||||||
|
_basePath.clear();
|
||||||
|
}
|
||||||
|
|
||||||
void ProviderRestApi::setPath(const QStringList& pathElements)
|
void ProviderRestApi::setPath(const QStringList& pathElements)
|
||||||
{
|
{
|
||||||
_path.clear();
|
_path.clear();
|
||||||
@ -83,6 +105,11 @@ void ProviderRestApi::setPath(const QString& path)
|
|||||||
appendPath(_path, path);
|
appendPath(_path, path);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ProviderRestApi::clearPath()
|
||||||
|
{
|
||||||
|
_path.clear();
|
||||||
|
}
|
||||||
|
|
||||||
void ProviderRestApi::appendPath(const QString& path)
|
void ProviderRestApi::appendPath(const QString& path)
|
||||||
{
|
{
|
||||||
appendPath(_path, path);
|
appendPath(_path, path);
|
||||||
@ -204,6 +231,7 @@ httpResponse ProviderRestApi::executeOperation(QNetworkAccessManager::Operation
|
|||||||
QDateTime start = QDateTime::currentDateTime();
|
QDateTime start = QDateTime::currentDateTime();
|
||||||
QString opCode;
|
QString opCode;
|
||||||
QNetworkReply* reply;
|
QNetworkReply* reply;
|
||||||
|
|
||||||
switch (operation) {
|
switch (operation) {
|
||||||
case QNetworkAccessManager::GetOperation:
|
case QNetworkAccessManager::GetOperation:
|
||||||
opCode = "GET";
|
opCode = "GET";
|
||||||
@ -255,11 +283,11 @@ httpResponse ProviderRestApi::getResponse(QNetworkReply* const& reply)
|
|||||||
HttpStatusCode httpStatusCode = static_cast<HttpStatusCode>(reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt());
|
HttpStatusCode httpStatusCode = static_cast<HttpStatusCode>(reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt());
|
||||||
response.setHttpStatusCode(httpStatusCode);
|
response.setHttpStatusCode(httpStatusCode);
|
||||||
response.setNetworkReplyError(reply->error());
|
response.setNetworkReplyError(reply->error());
|
||||||
|
response.setHeaders(reply->rawHeaderPairs());
|
||||||
|
|
||||||
if (reply->error() == QNetworkReply::NoError)
|
if (reply->error() == QNetworkReply::NoError)
|
||||||
{
|
{
|
||||||
QByteArray replyData = reply->readAll();
|
QByteArray replyData = reply->readAll();
|
||||||
|
|
||||||
if (!replyData.isEmpty())
|
if (!replyData.isEmpty())
|
||||||
{
|
{
|
||||||
QJsonParseError error;
|
QJsonParseError error;
|
||||||
@ -284,6 +312,13 @@ httpResponse ProviderRestApi::getResponse(QNetworkReply* const& reply)
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
QString errorReason;
|
QString errorReason;
|
||||||
|
if (reply->error() == QNetworkReply::OperationCanceledError)
|
||||||
|
{
|
||||||
|
errorReason = "Network request timeout error";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
qDebug() << "httpStatusCode: "<< httpStatusCode;
|
||||||
if (httpStatusCode > 0) {
|
if (httpStatusCode > 0) {
|
||||||
QString httpReason = reply->attribute(QNetworkRequest::HttpReasonPhraseAttribute).toString();
|
QString httpReason = reply->attribute(QNetworkRequest::HttpReasonPhraseAttribute).toString();
|
||||||
QString advise;
|
QString advise;
|
||||||
@ -307,17 +342,11 @@ httpResponse ProviderRestApi::getResponse(QNetworkReply* const& reply)
|
|||||||
errorReason = QString ("[%3 %4] - %5").arg(httpStatusCode).arg(httpReason, advise);
|
errorReason = QString ("[%3 %4] - %5").arg(httpStatusCode).arg(httpReason, advise);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
|
||||||
if (reply->error() == QNetworkReply::OperationCanceledError)
|
|
||||||
{
|
|
||||||
errorReason = "Network request timeout error";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
errorReason = reply->errorString();
|
errorReason = reply->errorString();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
response.setError(true);
|
response.setError(true);
|
||||||
response.setErrorReason(errorReason);
|
response.setErrorReason(errorReason);
|
||||||
}
|
}
|
||||||
@ -344,3 +373,121 @@ void ProviderRestApi::setHeader(const QByteArray &headerName, const QByteArray &
|
|||||||
{
|
{
|
||||||
_networkRequestHeaders.setRawHeader(headerName, headerValue);
|
_networkRequestHeaders.setRawHeader(headerName, headerValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void httpResponse::setHeaders(const QList<QNetworkReply::RawHeaderPair>& pairs)
|
||||||
|
{
|
||||||
|
_responseHeaders.clear();
|
||||||
|
for (const auto &item: pairs)
|
||||||
|
{
|
||||||
|
_responseHeaders[item.first] = item.second;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QByteArray httpResponse::getHeader(const QByteArray header) const
|
||||||
|
{
|
||||||
|
return _responseHeaders.value(header);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ProviderRestApi::setCaCertificate(const QString& caFileName)
|
||||||
|
{
|
||||||
|
bool rc {false};
|
||||||
|
/// Add our own CA to the default SSL configuration
|
||||||
|
QSslConfiguration configuration = QSslConfiguration::defaultConfiguration();
|
||||||
|
|
||||||
|
QFile caFile (caFileName);
|
||||||
|
if (!caFile.open(QIODevice::ReadOnly))
|
||||||
|
{
|
||||||
|
Error(_log,"Unable to open CA-Certificate file: %s", QSTRING_CSTR(caFileName));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
QSslCertificate cert (&caFile);
|
||||||
|
caFile.close();
|
||||||
|
|
||||||
|
QList<QSslCertificate> allowedCAs;
|
||||||
|
allowedCAs << cert;
|
||||||
|
configuration.setCaCertificates(allowedCAs);
|
||||||
|
|
||||||
|
QSslConfiguration::setDefaultConfiguration(configuration);
|
||||||
|
|
||||||
|
#ifndef QT_NO_SSL
|
||||||
|
if (QSslSocket::supportsSsl())
|
||||||
|
{
|
||||||
|
QObject::connect( _networkManager, &QNetworkAccessManager::sslErrors, this, &ProviderRestApi::onSslErrors, Qt::UniqueConnection );
|
||||||
|
_networkManager->connectToHostEncrypted(_apiUrl.host(), _apiUrl.port(), configuration);
|
||||||
|
rc = true;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProviderRestApi::acceptSelfSignedCertificates(bool isAccepted)
|
||||||
|
{
|
||||||
|
_isSeflSignedCertificateAccpeted = isAccepted;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProviderRestApi::setAlternateServerIdentity(const QString& serverIdentity)
|
||||||
|
{
|
||||||
|
_serverIdentity = serverIdentity;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString ProviderRestApi::getAlternateServerIdentity() const
|
||||||
|
{
|
||||||
|
return _serverIdentity;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ProviderRestApi::checkServerIdentity(const QSslConfiguration& sslConfig) const
|
||||||
|
{
|
||||||
|
bool isServerIdentified {false};
|
||||||
|
|
||||||
|
// Perform common name validation
|
||||||
|
QSslCertificate serverCertificate = sslConfig.peerCertificate();
|
||||||
|
QStringList commonName = serverCertificate.subjectInfo(QSslCertificate::CommonName);
|
||||||
|
if ( commonName.contains(getAlternateServerIdentity(), Qt::CaseInsensitive) )
|
||||||
|
{
|
||||||
|
isServerIdentified = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return isServerIdentified;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProviderRestApi::onSslErrors(QNetworkReply* reply, const QList<QSslError>& errors)
|
||||||
|
{
|
||||||
|
int ignoredErrorCount {0};
|
||||||
|
for (const QSslError &error : errors)
|
||||||
|
{
|
||||||
|
bool ignoreSslError{false};
|
||||||
|
|
||||||
|
switch (error.error()) {
|
||||||
|
case QSslError::HostNameMismatch :
|
||||||
|
if (checkServerIdentity(reply->sslConfiguration()) )
|
||||||
|
{
|
||||||
|
ignoreSslError = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case QSslError::SelfSignedCertificate :
|
||||||
|
if (_isSeflSignedCertificateAccpeted)
|
||||||
|
{
|
||||||
|
ignoreSslError = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ignoreSslError)
|
||||||
|
{
|
||||||
|
++ignoredErrorCount;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Debug (_log,"SSL Error occured: [%d] %s ",error.error(), QSTRING_CSTR(error.errorString()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ignoredErrorCount == errors.size())
|
||||||
|
{
|
||||||
|
reply->ignoreSslErrors();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -10,12 +10,13 @@
|
|||||||
#include <QUrlQuery>
|
#include <QUrlQuery>
|
||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
|
|
||||||
|
#include <QFile>
|
||||||
#include <QBasicTimer>
|
#include <QBasicTimer>
|
||||||
#include <QTimerEvent>
|
#include <QTimerEvent>
|
||||||
|
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
|
|
||||||
constexpr std::chrono::milliseconds DEFAULT_REST_TIMEOUT{ 1000 };
|
constexpr std::chrono::milliseconds DEFAULT_REST_TIMEOUT{ 2000 };
|
||||||
|
|
||||||
//Set QNetworkReply timeout without external timer
|
//Set QNetworkReply timeout without external timer
|
||||||
//https://stackoverflow.com/questions/37444539/how-to-set-qnetworkreply-timeout-without-external-timer
|
//https://stackoverflow.com/questions/37444539/how-to-set-qnetworkreply-timeout-without-external-timer
|
||||||
@ -87,6 +88,10 @@ public:
|
|||||||
QJsonDocument getBody() const { return _responseBody; }
|
QJsonDocument getBody() const { return _responseBody; }
|
||||||
void setBody(const QJsonDocument& body) { _responseBody = body; }
|
void setBody(const QJsonDocument& body) { _responseBody = body; }
|
||||||
|
|
||||||
|
|
||||||
|
QByteArray getHeader(const QByteArray header) const;
|
||||||
|
void setHeaders(const QList<QNetworkReply::RawHeaderPair>& pairs);
|
||||||
|
|
||||||
QString getErrorReason() const { return _errorReason; }
|
QString getErrorReason() const { return _errorReason; }
|
||||||
void setErrorReason(const QString& errorReason) { _errorReason = errorReason; }
|
void setErrorReason(const QString& errorReason) { _errorReason = errorReason; }
|
||||||
|
|
||||||
@ -99,6 +104,8 @@ public:
|
|||||||
private:
|
private:
|
||||||
|
|
||||||
QJsonDocument _responseBody {};
|
QJsonDocument _responseBody {};
|
||||||
|
QHash<QByteArray, QByteArray> _responseHeaders;
|
||||||
|
|
||||||
bool _hasError = false;
|
bool _hasError = false;
|
||||||
QString _errorReason;
|
QString _errorReason;
|
||||||
|
|
||||||
@ -131,6 +138,7 @@ class ProviderRestApi : public QObject
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
|
///
|
||||||
/// @brief Constructor of the REST-API wrapper
|
/// @brief Constructor of the REST-API wrapper
|
||||||
///
|
///
|
||||||
ProviderRestApi();
|
ProviderRestApi();
|
||||||
@ -176,6 +184,20 @@ public:
|
|||||||
///
|
///
|
||||||
virtual ~ProviderRestApi() override;
|
virtual ~ProviderRestApi() override;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Set the API's scheme
|
||||||
|
///
|
||||||
|
/// @param[in] scheme
|
||||||
|
///
|
||||||
|
void setScheme(const QString& scheme);
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Get the API's scheme
|
||||||
|
///
|
||||||
|
/// return schme
|
||||||
|
///
|
||||||
|
QString getScheme() { return _apiUrl.scheme(); }
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Set an API's host
|
/// @brief Set an API's host
|
||||||
///
|
///
|
||||||
@ -190,6 +212,13 @@ public:
|
|||||||
///
|
///
|
||||||
void setPort(const int port) { _apiUrl.setPort(port); }
|
void setPort(const int port) { _apiUrl.setPort(port); }
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Get the API's port
|
||||||
|
///
|
||||||
|
/// return port
|
||||||
|
///
|
||||||
|
int getPort() { return _apiUrl.port(); }
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Set an API's url
|
/// @brief Set an API's url
|
||||||
///
|
///
|
||||||
@ -204,6 +233,13 @@ public:
|
|||||||
///
|
///
|
||||||
QUrl getUrl() const;
|
QUrl getUrl() const;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Set an API's base path (the stable path element before addressing resources)
|
||||||
|
///
|
||||||
|
/// @param[in] pathElements to form a path, e.g. (clip,v2,resource) results in "/clip/v2/resource"
|
||||||
|
///
|
||||||
|
void setBasePath(const QStringList& pathElements);
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Set an API's base path (the stable path element before addressing resources)
|
/// @brief Set an API's base path (the stable path element before addressing resources)
|
||||||
///
|
///
|
||||||
@ -211,6 +247,11 @@ public:
|
|||||||
///
|
///
|
||||||
void setBasePath(const QString& basePath);
|
void setBasePath(const QString& basePath);
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Clear an API's base path (the stable path element before addressing resources)
|
||||||
|
///
|
||||||
|
void clearBasePath();
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Set an API's path to address resources
|
/// @brief Set an API's path to address resources
|
||||||
///
|
///
|
||||||
@ -218,12 +259,18 @@ public:
|
|||||||
///
|
///
|
||||||
void setPath(const QString& path);
|
void setPath(const QString& path);
|
||||||
|
|
||||||
|
///
|
||||||
/// @brief Set an API's path to address resources
|
/// @brief Set an API's path to address resources
|
||||||
///
|
///
|
||||||
/// @param[in] pathElements to form a path, e.g. (lights,1,state) results in "/lights/1/state/"
|
/// @param[in] pathElements to form a path, e.g. (lights,1,state) results in "/lights/1/state/"
|
||||||
///
|
///
|
||||||
void setPath(const QStringList& pathElements);
|
void setPath(const QStringList& pathElements);
|
||||||
|
|
||||||
|
///
|
||||||
|
/// @brief Clear an API's path
|
||||||
|
///
|
||||||
|
void clearPath();
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Append an API's path element to path set before
|
/// @brief Append an API's path element to path set before
|
||||||
///
|
///
|
||||||
@ -252,6 +299,10 @@ public:
|
|||||||
///
|
///
|
||||||
void setQuery(const QUrlQuery& query);
|
void setQuery(const QUrlQuery& query);
|
||||||
|
|
||||||
|
|
||||||
|
QString getBasePath() {return _basePath;}
|
||||||
|
QString getPath() {return _path;}
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Execute GET request
|
/// @brief Execute GET request
|
||||||
///
|
///
|
||||||
@ -359,6 +410,14 @@ public:
|
|||||||
/// @param[in] timeout in milliseconds.
|
/// @param[in] timeout in milliseconds.
|
||||||
void setTransferTimeout(std::chrono::milliseconds timeout = DEFAULT_REST_TIMEOUT) { _requestTimeout = timeout; }
|
void setTransferTimeout(std::chrono::milliseconds timeout = DEFAULT_REST_TIMEOUT) { _requestTimeout = timeout; }
|
||||||
|
|
||||||
|
|
||||||
|
bool setCaCertificate(const QString& caFileName);
|
||||||
|
|
||||||
|
void acceptSelfSignedCertificates(bool accept);
|
||||||
|
|
||||||
|
void setAlternateServerIdentity(const QString& serverIdentity);
|
||||||
|
QString getAlternateServerIdentity() const;
|
||||||
|
|
||||||
///
|
///
|
||||||
/// @brief Set the common logger for LED-devices.
|
/// @brief Set the common logger for LED-devices.
|
||||||
///
|
///
|
||||||
@ -366,6 +425,10 @@ public:
|
|||||||
///
|
///
|
||||||
void setLogger(Logger* log) { _log = log; }
|
void setLogger(Logger* log) { _log = log; }
|
||||||
|
|
||||||
|
protected slots:
|
||||||
|
/// Handle the SSLErrors
|
||||||
|
void onSslErrors(QNetworkReply* reply, const QList<QSslError>& errors);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
///
|
///
|
||||||
@ -379,9 +442,11 @@ private:
|
|||||||
|
|
||||||
httpResponse executeOperation(QNetworkAccessManager::Operation op, const QUrl& url, const QByteArray& body = {});
|
httpResponse executeOperation(QNetworkAccessManager::Operation op, const QUrl& url, const QByteArray& body = {});
|
||||||
|
|
||||||
|
bool checkServerIdentity(const QSslConfiguration& sslConfig) const;
|
||||||
|
|
||||||
Logger* _log;
|
Logger* _log;
|
||||||
|
|
||||||
// QNetworkAccessManager object for sending REST-requests.
|
/// QNetworkAccessManager object for sending REST-requests.
|
||||||
QNetworkAccessManager* _networkManager;
|
QNetworkAccessManager* _networkManager;
|
||||||
std::chrono::milliseconds _requestTimeout;
|
std::chrono::milliseconds _requestTimeout;
|
||||||
|
|
||||||
@ -394,6 +459,9 @@ private:
|
|||||||
QUrlQuery _query;
|
QUrlQuery _query;
|
||||||
|
|
||||||
QNetworkRequest _networkRequestHeaders;
|
QNetworkRequest _networkRequestHeaders;
|
||||||
|
|
||||||
|
QString _serverIdentity;
|
||||||
|
bool _isSeflSignedCertificateAccpeted;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // PROVIDERRESTKAPI_H
|
#endif // PROVIDERRESTKAPI_H
|
||||||
|
@ -150,6 +150,11 @@ const int *ProviderUdpSSL::getCiphersuites() const
|
|||||||
return mbedtls_ssl_list_ciphersuites();
|
return mbedtls_ssl_list_ciphersuites();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ProviderUdpSSL::setPSKidentity(const QString& pskIdentity)
|
||||||
|
{
|
||||||
|
_psk_identity = pskIdentity;
|
||||||
|
}
|
||||||
|
|
||||||
bool ProviderUdpSSL::initNetwork()
|
bool ProviderUdpSSL::initNetwork()
|
||||||
{
|
{
|
||||||
if ((!_isDeviceReady || _streamPaused) && _streamReady)
|
if ((!_isDeviceReady || _streamPaused) && _streamReady)
|
||||||
@ -334,6 +339,11 @@ void ProviderUdpSSL::freeSSLConnection()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ProviderUdpSSL::writeBytes(QByteArray data, bool flush)
|
||||||
|
{
|
||||||
|
writeBytes(static_cast<uint>(data.size()), reinterpret_cast<unsigned char*>(data.data()), flush);
|
||||||
|
}
|
||||||
|
|
||||||
void ProviderUdpSSL::writeBytes(unsigned int size, const uint8_t* data, bool flush)
|
void ProviderUdpSSL::writeBytes(unsigned int size, const uint8_t* data, bool flush)
|
||||||
{
|
{
|
||||||
if (!_streamReady || _streamPaused)
|
if (!_streamReady || _streamPaused)
|
||||||
|
@ -100,6 +100,14 @@ protected:
|
|||||||
///
|
///
|
||||||
void stopConnection();
|
void stopConnection();
|
||||||
|
|
||||||
|
///
|
||||||
|
/// Writes the given bytes/bits to the UDP-device and sleeps the latch time to ensure that the
|
||||||
|
/// values are latched.
|
||||||
|
///
|
||||||
|
/// @param[in] data The data
|
||||||
|
///
|
||||||
|
void writeBytes(QByteArray data, bool flush = false);
|
||||||
|
|
||||||
///
|
///
|
||||||
/// Writes the given bytes/bits to the UDP-device and sleeps the latch time to ensure that the
|
/// Writes the given bytes/bits to the UDP-device and sleeps the latch time to ensure that the
|
||||||
/// values are latched.
|
/// values are latched.
|
||||||
@ -116,6 +124,8 @@ protected:
|
|||||||
///
|
///
|
||||||
virtual const int * getCiphersuites() const;
|
virtual const int * getCiphersuites() const;
|
||||||
|
|
||||||
|
void setPSKidentity(const QString& pskIdentity);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
bool initConnection();
|
bool initConnection();
|
||||||
|
@ -35,26 +35,45 @@
|
|||||||
},
|
},
|
||||||
"propertyOrder": 4
|
"propertyOrder": 4
|
||||||
},
|
},
|
||||||
|
"useAPIv2": {
|
||||||
|
"type": "boolean",
|
||||||
|
"format": "checkbox",
|
||||||
|
"title": "edt_dev_spec_useAPIv2_title",
|
||||||
|
"default": false,
|
||||||
|
"options": {
|
||||||
|
"hidden": true
|
||||||
|
},
|
||||||
|
"access": "expert",
|
||||||
|
"propertyOrder": 5
|
||||||
|
},
|
||||||
"useEntertainmentAPI": {
|
"useEntertainmentAPI": {
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"format": "checkbox",
|
"format": "checkbox",
|
||||||
"title": "edt_dev_spec_useEntertainmentAPI_title",
|
"title": "edt_dev_spec_useEntertainmentAPI_title",
|
||||||
"default": true,
|
"default": true,
|
||||||
"propertyOrder": 5
|
"options": {
|
||||||
|
"hidden": true
|
||||||
|
},
|
||||||
|
"propertyOrder": 6
|
||||||
},
|
},
|
||||||
"switchOffOnBlack": {
|
"switchOffOnBlack": {
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"format": "checkbox",
|
"format": "checkbox",
|
||||||
"title": "edt_dev_spec_switchOffOnBlack_title",
|
"title": "edt_dev_spec_switchOffOnBlack_title",
|
||||||
"default": false,
|
"default": false,
|
||||||
"propertyOrder": 6
|
"options": {
|
||||||
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"propertyOrder": 7
|
||||||
},
|
},
|
||||||
"restoreOriginalState": {
|
"restoreOriginalState": {
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"format": "checkbox",
|
"format": "checkbox",
|
||||||
"title": "edt_dev_spec_restoreOriginalState_title",
|
"title": "edt_dev_spec_restoreOriginalState_title",
|
||||||
"default": false,
|
"default": false,
|
||||||
"propertyOrder": 7
|
"propertyOrder": 8
|
||||||
},
|
},
|
||||||
"blackLevel": {
|
"blackLevel": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
@ -64,7 +83,12 @@
|
|||||||
"step": 0.01,
|
"step": 0.01,
|
||||||
"minimum": 0.001,
|
"minimum": 0.001,
|
||||||
"maximum": 1.0,
|
"maximum": 1.0,
|
||||||
"propertyOrder": 8
|
"options": {
|
||||||
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"propertyOrder": 9
|
||||||
},
|
},
|
||||||
"onBlackTimeToPowerOff": {
|
"onBlackTimeToPowerOff": {
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
@ -76,7 +100,12 @@
|
|||||||
"maximum": 100000,
|
"maximum": 100000,
|
||||||
"default": 600,
|
"default": 600,
|
||||||
"required": true,
|
"required": true,
|
||||||
"propertyOrder": 9
|
"options": {
|
||||||
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"propertyOrder": 10
|
||||||
},
|
},
|
||||||
"onBlackTimeToPowerOn": {
|
"onBlackTimeToPowerOn": {
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
@ -88,14 +117,24 @@
|
|||||||
"maximum": 100000,
|
"maximum": 100000,
|
||||||
"default": 300,
|
"default": 300,
|
||||||
"required": true,
|
"required": true,
|
||||||
"propertyOrder": 9
|
"options": {
|
||||||
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"propertyOrder": 11
|
||||||
},
|
},
|
||||||
"candyGamma": {
|
"candyGamma": {
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"format": "checkbox",
|
"format": "checkbox",
|
||||||
"title": "edt_dev_spec_candyGamma_title",
|
"title": "edt_dev_spec_candyGamma_title",
|
||||||
"default": true,
|
"default": true,
|
||||||
"propertyOrder": 10
|
"options": {
|
||||||
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"propertyOrder": 12
|
||||||
},
|
},
|
||||||
"lightIds": {
|
"lightIds": {
|
||||||
"type": "array",
|
"type": "array",
|
||||||
@ -112,20 +151,23 @@
|
|||||||
"useEntertainmentAPI": false
|
"useEntertainmentAPI": false
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 11
|
"propertyOrder": 13
|
||||||
},
|
},
|
||||||
"groupId": {
|
"groupId": {
|
||||||
"type": "number",
|
"type": "string",
|
||||||
"format": "stepper",
|
|
||||||
"step": 1,
|
|
||||||
"title": "edt_dev_spec_groupId_title",
|
"title": "edt_dev_spec_groupId_title",
|
||||||
"default": 0,
|
"default": "",
|
||||||
"options": {
|
"options": {
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"useEntertainmentAPI": true
|
"useEntertainmentAPI": true
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 12
|
"options": {
|
||||||
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"propertyOrder": 14
|
||||||
},
|
},
|
||||||
"brightnessFactor": {
|
"brightnessFactor": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
@ -136,7 +178,12 @@
|
|||||||
"minimum": 0.5,
|
"minimum": 0.5,
|
||||||
"maximum": 10.0,
|
"maximum": 10.0,
|
||||||
"access": "advanced",
|
"access": "advanced",
|
||||||
"propertyOrder": 13
|
"options": {
|
||||||
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"propertyOrder": 15
|
||||||
},
|
},
|
||||||
"handshakeTimeoutMin": {
|
"handshakeTimeoutMin": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
@ -154,7 +201,7 @@
|
|||||||
"useEntertainmentAPI": true
|
"useEntertainmentAPI": true
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 14
|
"propertyOrder": 16
|
||||||
},
|
},
|
||||||
"handshakeTimeoutMax": {
|
"handshakeTimeoutMax": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
@ -172,7 +219,7 @@
|
|||||||
"useEntertainmentAPI": true
|
"useEntertainmentAPI": true
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 15
|
"propertyOrder": 17
|
||||||
},
|
},
|
||||||
"verbose": {
|
"verbose": {
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
@ -180,7 +227,7 @@
|
|||||||
"title": "edt_dev_spec_verbose_title",
|
"title": "edt_dev_spec_verbose_title",
|
||||||
"default": false,
|
"default": false,
|
||||||
"access": "expert",
|
"access": "expert",
|
||||||
"propertyOrder": 16
|
"propertyOrder": 18
|
||||||
},
|
},
|
||||||
"transitiontime": {
|
"transitiontime": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
@ -195,24 +242,29 @@
|
|||||||
"useEntertainmentAPI": false
|
"useEntertainmentAPI": false
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 17
|
"propertyOrder": 19
|
||||||
},
|
},
|
||||||
"blackLightsTimeout": {
|
"blackLightsTimeout": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
|
"title": "edt_dev_spec_blackLightsTimeout_title",
|
||||||
"default": 5000,
|
"default": 5000,
|
||||||
"options": {
|
"options": {
|
||||||
"hidden": true
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 18
|
"propertyOrder": 20
|
||||||
},
|
},
|
||||||
"brightnessThreshold": {
|
"brightnessThreshold": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
"title": "edt_dev_spec_brightnessThreshold_title",
|
"title": "edt_dev_spec_brightnessThreshold_title",
|
||||||
"default": 0.0001,
|
"default": 0.0001,
|
||||||
"options": {
|
"options": {
|
||||||
"hidden": true
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 19
|
"propertyOrder": 21
|
||||||
},
|
},
|
||||||
"brightnessMin": {
|
"brightnessMin": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
@ -223,9 +275,11 @@
|
|||||||
"maximum": 1.0,
|
"maximum": 1.0,
|
||||||
"access": "advanced",
|
"access": "advanced",
|
||||||
"options": {
|
"options": {
|
||||||
"hidden": true
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 20
|
"propertyOrder": 22
|
||||||
},
|
},
|
||||||
"brightnessMax": {
|
"brightnessMax": {
|
||||||
"type": "number",
|
"type": "number",
|
||||||
@ -236,9 +290,11 @@
|
|||||||
"maximum": 1.0,
|
"maximum": 1.0,
|
||||||
"access": "advanced",
|
"access": "advanced",
|
||||||
"options": {
|
"options": {
|
||||||
"hidden": true
|
"dependencies": {
|
||||||
|
"useAPIv2": false
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"propertyOrder": 21
|
"propertyOrder": 23
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"additionalProperties": true
|
"additionalProperties": true
|
||||||
|
@ -182,6 +182,7 @@ bool MdnsBrowser::resolveAddress(Logger* log, const QString& hostname, QHostAddr
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
QObject::disconnect(&MdnsBrowser::getInstance(), &MdnsBrowser::addressResolved, nullptr, nullptr);
|
||||||
Error(log, "Resolved mDNS hostname [%s] timed out", QSTRING_CSTR(hostname));
|
Error(log, "Resolved mDNS hostname [%s] timed out", QSTRING_CSTR(hostname));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
14
resources/ssl/philips_hue_ca.pem
Normal file
14
resources/ssl/philips_hue_ca.pem
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
-----BEGIN CERTIFICATE-----
|
||||||
|
MIICMjCCAdigAwIBAgIUO7FSLbaxikuXAljzVaurLXWmFw4wCgYIKoZIzj0EAwIw
|
||||||
|
OTELMAkGA1UEBhMCTkwxFDASBgNVBAoMC1BoaWxpcHMgSHVlMRQwEgYDVQQDDAty
|
||||||
|
b290LWJyaWRnZTAiGA8yMDE3MDEwMTAwMDAwMFoYDzIwMzgwMTE5MDMxNDA3WjA5
|
||||||
|
MQswCQYDVQQGEwJOTDEUMBIGA1UECgwLUGhpbGlwcyBIdWUxFDASBgNVBAMMC3Jv
|
||||||
|
b3QtYnJpZGdlMFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEjNw2tx2AplOf9x86
|
||||||
|
aTdvEcL1FU65QDxziKvBpW9XXSIcibAeQiKxegpq8Exbr9v6LBnYbna2VcaK0G22
|
||||||
|
jOKkTqOBuTCBtjAPBgNVHRMBAf8EBTADAQH/MA4GA1UdDwEB/wQEAwIBhjAdBgNV
|
||||||
|
HQ4EFgQUZ2ONTFrDT6o8ItRnKfqWKnHFGmQwdAYDVR0jBG0wa4AUZ2ONTFrDT6o8
|
||||||
|
ItRnKfqWKnHFGmShPaQ7MDkxCzAJBgNVBAYTAk5MMRQwEgYDVQQKDAtQaGlsaXBz
|
||||||
|
IEh1ZTEUMBIGA1UEAwwLcm9vdC1icmlkZ2WCFDuxUi22sYpLlwJY81Wrqy11phcO
|
||||||
|
MAoGCCqGSM49BAMCA0gAMEUCIEBYYEOsa07TH7E5MJnGw557lVkORgit2Rm1h3B2
|
||||||
|
sFgDAiEA1Fj/C3AN5psFMjo0//mrQebo0eKd3aWRx+pQY08mk48=
|
||||||
|
-----END CERTIFICATE-----
|
Loading…
x
Reference in New Issue
Block a user