[outputs] Add a more standard device authentication function

outputs_authorize() has two issues, one that the caller can't specify device
(problem if there are two devices waiting for verification), the other that it
didn't offer a standard callback, so difficult to catch failure/success.
This commit is contained in:
ejurgensen 2020-05-26 22:42:06 +02:00
parent 9308f81224
commit 2fa2d33602
3 changed files with 81 additions and 27 deletions

View File

@ -942,6 +942,19 @@ outputs_device_quality_set(struct output_device *device, struct media_quality *q
return device_state_update(device, ret); return device_state_update(device, ret);
} }
int
outputs_device_authorize(struct output_device *device, const char *pin, output_status_cb cb)
{
int ret;
if (outputs[device->type]->disabled || !outputs[device->type]->device_authorize)
return -1;
ret = outputs[device->type]->device_authorize(device, pin, callback_add(device, cb));
return device_state_update(device, ret);
}
void void
outputs_device_cb_set(struct output_device *device, output_status_cb cb) outputs_device_cb_set(struct output_device *device, output_status_cb cb)
{ {

View File

@ -226,6 +226,9 @@ struct output_definition
// Request a change of quality from the device // Request a change of quality from the device
int (*device_quality_set)(struct output_device *device, struct media_quality *quality, int callback_id); int (*device_quality_set)(struct output_device *device, struct media_quality *quality, int callback_id);
// Authorize forked-daapd to use the device
int (*device_authorize)(struct output_device *device, const char *pin, int callback_id);
// Change the call back associated with a device // Change the call back associated with a device
void (*device_cb_set)(struct output_device *device, int callback_id); void (*device_cb_set)(struct output_device *device, int callback_id);
@ -319,6 +322,9 @@ outputs_device_volume_to_pct(struct output_device *device, const char *value);
int int
outputs_device_quality_set(struct output_device *device, struct media_quality *quality, output_status_cb cb); outputs_device_quality_set(struct output_device *device, struct media_quality *quality, output_status_cb cb);
int
outputs_device_authorize(struct output_device *device, const char *pin, output_status_cb cb);
void void
outputs_device_cb_set(struct output_device *device, output_status_cb cb); outputs_device_cb_set(struct output_device *device, output_status_cb cb);

View File

@ -4278,20 +4278,20 @@ raop_cb_verification_setup_step3(struct evrtsp_request *req, void *arg)
ret = raop_verification_response_process(3, req, rs); ret = raop_verification_response_process(3, req, rs);
if (ret < 0) if (ret < 0)
goto error; goto out;
ret = verification_setup_result(&authorization_key, rs->verification_setup_ctx); ret = verification_setup_result(&authorization_key, rs->verification_setup_ctx);
if (ret < 0) if (ret < 0)
{ {
DPRINTF(E_LOG, L_RAOP, "Verification setup result error: %s\n", verification_setup_errmsg(rs->verification_setup_ctx)); DPRINTF(E_LOG, L_RAOP, "Verification setup result error: %s\n", verification_setup_errmsg(rs->verification_setup_ctx));
goto error; goto out;
} }
DPRINTF(E_LOG, L_RAOP, "Verification setup stage complete, saving authorization key\n"); DPRINTF(E_LOG, L_RAOP, "Verification setup stage complete, saving authorization key\n");
device = outputs_device_get(rs->device_id); device = outputs_device_get(rs->device_id);
if (!device) if (!device)
goto error; goto out;
free(device->auth_key); free(device->auth_key);
device->auth_key = strdup(authorization_key); device->auth_key = strdup(authorization_key);
@ -4299,14 +4299,19 @@ raop_cb_verification_setup_step3(struct evrtsp_request *req, void *arg)
// A blocking db call... :-~ // A blocking db call... :-~
db_speaker_save(device); db_speaker_save(device);
// The player considers this session failed, so we don't need it any more // No longer UNVERIFIED
session_cleanup(rs); rs->state = RAOP_STATE_STOPPED;
/* Fallthrough */ out:
error:
verification_setup_free(rs->verification_setup_ctx); verification_setup_free(rs->verification_setup_ctx);
rs->verification_setup_ctx = NULL; rs->verification_setup_ctx = NULL;
// Callback to player with result
raop_status(rs);
// We are telling the player that the device is now stopped, so we don't need
// the session any more
session_cleanup(rs);
} }
static void static void
@ -4328,6 +4333,7 @@ raop_cb_verification_setup_step2(struct evrtsp_request *req, void *arg)
error: error:
verification_setup_free(rs->verification_setup_ctx); verification_setup_free(rs->verification_setup_ctx);
rs->verification_setup_ctx = NULL; rs->verification_setup_ctx = NULL;
raop_status(rs);
} }
static void static void
@ -4349,13 +4355,37 @@ raop_cb_verification_setup_step1(struct evrtsp_request *req, void *arg)
error: error:
verification_setup_free(rs->verification_setup_ctx); verification_setup_free(rs->verification_setup_ctx);
rs->verification_setup_ctx = NULL; rs->verification_setup_ctx = NULL;
raop_status(rs);
}
static int
raop_verification_setup(struct raop_session *rs, const char *pin)
{
int ret;
rs->verification_setup_ctx = verification_setup_new(pin);
if (!rs->verification_setup_ctx)
{
DPRINTF(E_LOG, L_RAOP, "Out of memory for verification setup context\n");
return -1;
}
ret = raop_verification_request_send(1, rs, raop_cb_verification_setup_step1);
if (ret < 0)
goto error;
return 0;
error:
verification_setup_free(rs->verification_setup_ctx);
rs->verification_setup_ctx = NULL;
return -1;
} }
static void static void
raop_verification_setup(const char *pin) raop_authorize(const char *pin)
{ {
struct raop_session *rs; struct raop_session *rs;
int ret;
for (rs = raop_sessions; rs; rs = rs->next) for (rs = raop_sessions; rs; rs = rs->next)
{ {
@ -4369,23 +4399,27 @@ raop_verification_setup(const char *pin)
return; return;
} }
rs->verification_setup_ctx = verification_setup_new(pin); raop_verification_setup(rs, pin);
if (!rs->verification_setup_ctx) }
static int
raop_device_authorize(struct output_device *device, const char *pin, int callback_id)
{ {
DPRINTF(E_LOG, L_RAOP, "Out of memory for verification setup context\n"); struct raop_session *rs = device->session;
return; int ret;
}
ret = raop_verification_request_send(1, rs, raop_cb_verification_setup_step1); if (!rs)
return -1;
ret = raop_verification_setup(rs, pin);
if (ret < 0) if (ret < 0)
goto error; return -1;
return; rs->callback_id = callback_id;
error: return 1;
verification_setup_free(rs->verification_setup_ctx);
rs->verification_setup_ctx = NULL;
} }
#else #else
static int static int
raop_verification_verify(struct raop_session *rs) raop_verification_verify(struct raop_session *rs)
@ -5011,6 +5045,7 @@ struct output_definition output_raop =
.metadata_send = raop_metadata_send, .metadata_send = raop_metadata_send,
.metadata_purge = raop_metadata_purge, .metadata_purge = raop_metadata_purge,
#ifdef RAOP_VERIFICATION #ifdef RAOP_VERIFICATION
.authorize = raop_verification_setup, .device_authorize = raop_device_authorize,
.authorize = raop_authorize,
#endif #endif
}; };