Merge branch 'nielsdg/extract-get-secret-file-func' into 'master'

file-backend: Extract functions from the initializer code

See merge request GNOME/libsecret!107
This commit is contained in:
Niels De Graef 2023-01-15 10:16:50 +00:00
commit 33a4de59a2

View File

@ -60,6 +60,51 @@ enum {
PROP_FLAGS PROP_FLAGS
}; };
/* Gets the GFile for this backend and makes sure the parent dirs exist */
static GFile *
get_secret_file (GCancellable *cancellable, GError **error)
{
const char *envvar = NULL;
char *path = NULL;
GFile *file = NULL;
GFile *dir = NULL;
gboolean ret;
envvar = g_getenv ("SECRET_FILE_TEST_PATH");
if (envvar != NULL && *envvar != '\0') {
path = g_strdup (envvar);
} else {
path = g_build_filename (g_get_user_data_dir (),
"keyrings",
SECRET_COLLECTION_DEFAULT ".keyring",
NULL);
}
file = g_file_new_for_path (path);
g_free (path);
dir = g_file_get_parent (file);
if (dir == NULL) {
g_set_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT,
"not a valid path");
g_object_unref (file);
return NULL;
}
ret = g_file_make_directory_with_parents (dir, cancellable, error);
g_object_unref (dir);
if (!ret) {
if (!g_error_matches (*error, G_IO_ERROR, G_IO_ERROR_EXISTS)) {
g_object_unref (file);
return NULL;
}
g_clear_error (error);
}
return file;
}
static void static void
secret_file_backend_init (SecretFileBackend *self) secret_file_backend_init (SecretFileBackend *self)
{ {
@ -419,6 +464,77 @@ on_bus_get (GObject *source_object,
g_object_unref (fd_list); g_object_unref (fd_list);
} }
#ifdef WITH_TPM
static GBytes *
load_password_from_tpm (GFile *file, GCancellable *cancellable, GError **error)
{
EggTpm2Context *context = NULL;
char *path = NULL;
char *tpm2_file_path = NULL;
GFile *tpm2_file = NULL;
gboolean status;
GBytes *encrypted = NULL;
GBytes *decrypted = NULL;
context = egg_tpm2_initialize (error);
if (!context)
return NULL;
path = g_file_get_path (file);
tpm2_file_path = g_strdup_printf ("%s.tpm2", path);
g_free (path);
tpm2_file = g_file_new_for_path (tpm2_file_path);
status = g_file_test (tpm2_file_path, G_FILE_TEST_EXISTS);
g_free (tpm2_file_path);
if (!status) {
gconstpointer contents;
gsize size;
encrypted = egg_tpm2_generate_master_password (context, error);
if (!encrypted)
return NULL;
contents = g_bytes_get_data (encrypted, &size);
status = g_file_replace_contents (tpm2_file,
contents,
size,
NULL,
FALSE,
G_FILE_CREATE_PRIVATE,
NULL,
cancellable,
error);
if (!status)
goto out;
} else {
char *contents;
gsize length;
status = g_file_load_contents (tpm2_file,
cancellable,
&contents,
&length,
NULL,
error);
if (!status)
goto out;
encrypted = g_bytes_new_take (contents, length);
}
decrypted = egg_tpm2_decrypt_master_password (context, encrypted, error);
out:
g_clear_object (&tpm2_file);
g_clear_pointer (&encrypted, g_bytes_unref);
egg_tpm2_finalize (context);
return decrypted;
}
#endif /* WITH_TPM */
static void static void
secret_file_backend_real_init_async (GAsyncInitable *initable, secret_file_backend_real_init_async (GAsyncInitable *initable,
int io_priority, int io_priority,
@ -426,55 +542,22 @@ secret_file_backend_real_init_async (GAsyncInitable *initable,
GAsyncReadyCallback callback, GAsyncReadyCallback callback,
gpointer user_data) gpointer user_data)
{ {
gchar *path; const char *envvar = NULL;
GFile *file; GFile *file = NULL;
GFile *dir;
SecretValue *password; SecretValue *password;
const gchar *envvar;
GTask *task; GTask *task;
GError *error = NULL; GError *error = NULL;
InitClosure *init; InitClosure *init;
gboolean ret;
task = g_task_new (initable, cancellable, callback, user_data); task = g_task_new (initable, cancellable, callback, user_data);
envvar = g_getenv ("SECRET_FILE_TEST_PATH"); file = get_secret_file (cancellable, &error);
if (envvar != NULL && *envvar != '\0') if (file == NULL) {
path = g_strdup (envvar); g_task_return_error (task, g_steal_pointer (&error));
else {
path = g_build_filename (g_get_user_data_dir (),
"keyrings",
SECRET_COLLECTION_DEFAULT ".keyring",
NULL);
}
file = g_file_new_for_path (path);
g_free (path);
dir = g_file_get_parent (file);
if (dir == NULL) {
g_task_return_new_error (task,
G_IO_ERROR,
G_IO_ERROR_INVALID_ARGUMENT,
"not a valid path");
g_object_unref (file);
g_object_unref (task); g_object_unref (task);
return; return;
} }
ret = g_file_make_directory_with_parents (dir, cancellable, &error);
g_object_unref (dir);
if (!ret) {
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_EXISTS))
g_clear_error (&error);
else {
g_task_return_error (task, error);
g_object_unref (file);
g_object_unref (task);
return;
}
}
envvar = g_getenv ("SECRET_FILE_TEST_PASSWORD"); envvar = g_getenv ("SECRET_FILE_TEST_PASSWORD");
if (envvar != NULL && *envvar != '\0') { if (envvar != NULL && *envvar != '\0') {
password = secret_value_new (envvar, -1, "text/plain"); password = secret_value_new (envvar, -1, "text/plain");
@ -496,90 +579,19 @@ secret_file_backend_real_init_async (GAsyncInitable *initable,
g_bus_get (G_BUS_TYPE_SESSION, cancellable, on_bus_get, task); g_bus_get (G_BUS_TYPE_SESSION, cancellable, on_bus_get, task);
} else { } else {
#ifdef WITH_TPM #ifdef WITH_TPM
EggTpm2Context *context; GBytes *decrypted = NULL;
GFile *tpm2_file; gconstpointer data;
gchar *tpm2_file_path;
gboolean status;
GBytes *encrypted;
GBytes *decrypted;
context = egg_tpm2_initialize (&error);
if (!context) {
g_task_return_error (task, error);
g_object_unref (task);
return;
}
path = g_file_get_path (file);
tpm2_file_path = g_strdup_printf ("%s.tpm2", path);
g_free(path);
tpm2_file = g_file_new_for_path (tpm2_file_path);
status = g_file_test (tpm2_file_path, G_FILE_TEST_EXISTS);
g_free (tpm2_file_path);
if (!status) {
encrypted = egg_tpm2_generate_master_password (
context,
&error);
if (!encrypted) {
g_task_return_error (task, error);
g_object_unref (task);
return;
}
gconstpointer contents;
gsize size; gsize size;
contents = g_bytes_get_data (encrypted, &size);
status = g_file_replace_contents (tpm2_file,
contents,
size,
NULL,
FALSE,
G_FILE_CREATE_PRIVATE,
NULL,
cancellable,
&error);
if (!status) {
g_task_return_error (task, error);
g_object_unref (task);
return;
}
} else { decrypted = load_password_from_tpm (file, cancellable, &error);
char *contents;
gsize length;
status = g_file_load_contents (tpm2_file,
cancellable,
&contents,
&length,
NULL,
&error);
if (!status) {
g_task_return_error (task, error);
g_object_unref (task);
return;
}
encrypted = g_bytes_new_take (contents, length);
}
decrypted = egg_tpm2_decrypt_master_password (context,
encrypted,
&error);
g_bytes_unref (encrypted);
egg_tpm2_finalize (context);
if (!decrypted) { if (!decrypted) {
g_task_return_error (task, error); g_task_return_error (task, error);
g_object_unref (task); g_object_unref (task);
return; return;
} }
gconstpointer data; data = g_bytes_get_data (decrypted, &size);
gsize size; password = secret_value_new (data,size, "text/plain");
data = g_bytes_get_data(decrypted, &size);
password = secret_value_new (data,
size,
"text/plain");
g_bytes_unref (decrypted); g_bytes_unref (decrypted);
g_async_initable_new_async (SECRET_TYPE_FILE_COLLECTION, g_async_initable_new_async (SECRET_TYPE_FILE_COLLECTION,
io_priority, io_priority,
@ -590,16 +602,15 @@ secret_file_backend_real_init_async (GAsyncInitable *initable,
"password", password, "password", password,
NULL); NULL);
g_object_unref (tpm2_file);
g_object_unref (file); g_object_unref (file);
secret_value_unref (password); secret_value_unref (password);
return;
#else #else
g_task_return_new_error (task, g_task_return_new_error (task,
G_IO_ERROR, G_IO_ERROR,
G_IO_ERROR_INVALID_ARGUMENT, G_IO_ERROR_INVALID_ARGUMENT,
"master password is not retrievable"); "master password is not retrievable");
g_object_unref (task); g_object_unref (task);
return;
#endif #endif
} }
} }