2025-02-26 21:29:42 +01:00

199 lines
5.7 KiB
Perl

package FHEM::Bluelink;
use strict;
use warnings;
use LWP::UserAgent;
use HTTP::Request::Common qw(GET POST);
use JSON;
use MIME::Base64;
use Digest::SHA qw(sha256_hex);
use Time::HiRes qw(time);
sub Initialize {
my ($hash) = @_;
$hash->{DefFn} = "Bluelink_Define";
$hash->{UndefFn} = "Bluelink_Undef";
$hash->{RenameFn} = "Bluelink_Rename";
$hash->{DeleteFn} = "Bluelink_Delete";
$hash->{NotifyFn} = "Bluelink_Notify";
$hash->{SetFn} = "Bluelink_Set";
$hash->{GetFn} = "Bluelink_Get";
$hash->{AttrList} = "username password language pin client_id client_secret access_token refresh_token";
}
sub Bluelink_Define {
my ($hash, $def) = @_;
my @args = split("[ \t]+", $def);
return "Usage: define <name> Bluelink <API_URL>" if (@args < 3);
$hash->{API_URL} = $args[2];
return;
}
sub Bluelink_Undef {
my ($hash) = @_;
return;
}
sub Bluelink_Set {
my ($hash, $name, $cmd, @args) = @_;
return "\"set $name\" needs at least one argument" unless defined $cmd;
if ($cmd eq "login") {
return Bluelink_Login($hash);
} elsif ($cmd eq "refreshToken") {
return Bluelink_RefreshToken($hash);
} elsif ($cmd eq "setLanguage") {
return Bluelink_SetLanguage($hash, $args[0]);
}
return "Unknown argument $cmd, choose one of login refreshToken setLanguage";
}
sub Bluelink_Get {
my ($hash, $name, $cmd, @args) = @_;
return "\"get $name\" needs at least one argument" unless defined $cmd;
if ($cmd eq "deviceid") {
return Bluelink_GetDeviceID($hash);
} elsif ($cmd eq "token") {
return AttrVal($name, "access_token", "No token available");
}
return "Unknown argument $cmd, choose one of deviceid token";
}
sub Bluelink_GetDeviceID {
my ($hash) = @_;
my $api_url = $hash->{API_URL} . "/api/v1/spa/notifications/register";
my $uuid = lc(sha256_hex(time() . rand()));
my $data = {
pushRegId => uc(unpack("H*", pack("C*", map { int(rand(16)) } (1..32)))),
pushType => "GCM",
uuid => $uuid
};
my $res = Bluelink_HttpPost($api_url, $data);
return "Device ID: " . ($res->{ResMsg}->{DeviceID} // "Not found");
}
sub Bluelink_Login {
my ($hash) = @_;
my $username = AttrVal($hash->{NAME}, "username", "");
my $password = AttrVal($hash->{NAME}, "password", "");
my $api_url = $hash->{API_URL} . "/api/v1/user/signin";
return "Missing username or password" unless $username && $password;
my $data = { email => $username, password => $password };
my $res = Bluelink_HttpPost($api_url, $data);
return "Login failed: " . $res->{errMsg} if $res->{errCode};
my $redirect_url = $res->{redirectUrl};
my $code = (split(/\?code=/, $redirect_url))[1];
return Bluelink_ExchangeCode($hash, $code);
}
sub Bluelink_ExchangeCode {
my ($hash, $code) = @_;
my $api_url = $hash->{API_URL} . "/api/v1/user/oauth2/token";
my $client_id = AttrVal($hash->{NAME}, "client_id", "");
my $client_secret = AttrVal($hash->{NAME}, "client_secret", "");
return "Missing client_id or client_secret" unless $client_id && $client_secret;
my $auth = encode_base64("$client_id:$client_secret", "");
my $data = {
grant_type => "authorization_code",
redirect_uri => $hash->{API_URL} . "/api/v1/user/oauth2/redirect",
code => $code
};
my $res = Bluelink_HttpPost($api_url, $data, { Authorization => "Basic $auth" });
if ($res->{access_token}) {
fhem("attr $hash->{NAME} access_token " . $res->{access_token});
fhem("attr $hash->{NAME} refresh_token " . $res->{refresh_token});
return "Login successful. Token saved.";
}
return "Token exchange failed.";
}
sub Bluelink_RefreshToken {
my ($hash) = @_;
my $api_url = $hash->{API_URL} . "/api/v1/user/oauth2/token";
my $refresh_token = AttrVal($hash->{NAME}, "refresh_token", "");
my $client_id = AttrVal($hash->{NAME}, "client_id", "");
my $client_secret = AttrVal($hash->{NAME}, "client_secret", "");
return "Missing refresh_token, client_id or client_secret" unless $refresh_token && $client_id && $client_secret;
my $auth = encode_base64("$client_id:$client_secret", "");
my $data = {
grant_type => "refresh_token",
redirect_uri => "https://www.getpostman.com/oauth2/callback",
refresh_token => $refresh_token
};
my $res = Bluelink_HttpPost($api_url, $data, { Authorization => "Basic $auth" });
if ($res->{access_token}) {
fhem("attr $hash->{NAME} access_token " . $res->{access_token});
fhem("attr $hash->{NAME} refresh_token " . $res->{refresh_token});
return "Token refreshed.";
}
return "Token refresh failed.";
}
sub Bluelink_SetLanguage {
my ($hash, $language) = @_;
my $api_url = $hash->{API_URL} . "/api/v1/user/language";
my $data = { lang => $language };
my $res = Bluelink_HttpPost($api_url, $data);
return "Language set to $language" if $res;
return "Failed to set language.";
}
sub Bluelink_HttpPost {
my ($url, $data, $headers) = @_;
my $ua = LWP::UserAgent->new();
my $json_data = encode_json($data);
my $req = HTTP::Request->new(POST => $url);
$req->header("Content-Type" => "application/json");
$req->content($json_data);
if ($headers) {
foreach my $key (keys %$headers) {
$req->header($key => $headers->{$key});
}
}
my $res = $ua->request($req);
return decode_json($res->decoded_content) if $res->is_success;
return { error => "Request failed", status => $res->code };
}
1;