2017-09-22 12:49:30 +01:00
/*
2017-09-21 20:22:09 +01:00
* Copyright © 2017 Endless Mobile , Inc .
*
2018-01-30 20:26:26 +01:00
* SPDX - License - Identifier : LGPL - 2.0 +
*
2017-09-21 20:22:09 +01:00
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
* version 2 of the License , or ( at your option ) any later version .
*
* This library is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public
2021-12-06 20:20:55 -05:00
* License along with this library . If not , see < https : //www.gnu.org/licenses/>.
2017-09-21 20:22:09 +01:00
*
* Authors :
* - Philip Withnall < withnall @ endlessm . com >
*/
# include "config.h"
# include <gio/gio.h>
# include <glib-object.h>
2023-05-01 14:24:29 -04:00
# include <glib.h>
2017-09-21 20:22:09 +01:00
# include <libglnx.h>
# include <locale.h>
# include "ostree-autocleanups.h"
# include "ostree-types.h"
/* Test fixture. Creates a temporary directory. */
typedef struct
{
2023-05-01 14:24:29 -04:00
GLnxTmpDir tmpdir ; /* (owned) */
2017-09-21 20:22:09 +01:00
} Fixture ;
static void
2023-05-01 14:24:29 -04:00
setup ( Fixture * fixture , gconstpointer test_data )
2017-09-21 20:22:09 +01:00
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
2017-09-21 20:22:09 +01:00
2023-05-01 14:24:29 -04:00
( void ) glnx_mkdtemp ( " test-repo-XXXXXX " , 0700 , & fixture - > tmpdir , & error ) ;
2017-09-21 20:22:09 +01:00
g_assert_no_error ( error ) ;
g_test_message ( " Using temporary directory: %s " , fixture - > tmpdir . path ) ;
}
2021-04-22 09:10:15 -06:00
/* Common setup for locking tests. Create an archive repo in the tmpdir and
* set the locking timeout to 0 so lock failures don ' t block .
*/
static void
2023-05-01 14:24:29 -04:00
lock_setup ( Fixture * fixture , gconstpointer test_data )
2021-04-22 09:10:15 -06:00
{
setup ( fixture , test_data ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
g_autoptr ( OstreeRepo ) repo = ostree_repo_create_at (
fixture - > tmpdir . fd , " . " , OSTREE_REPO_MODE_ARCHIVE , NULL , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
/* Set the lock timeout to 0 so failures don't block the test */
2023-05-01 14:24:29 -04:00
g_autoptr ( GKeyFile ) config = ostree_repo_copy_config ( repo ) ;
2021-04-22 09:10:15 -06:00
g_key_file_set_integer ( config , " core " , " lock-timeout-secs " , 0 ) ;
ostree_repo_write_config ( repo , config , & error ) ;
g_assert_no_error ( error ) ;
}
2017-09-21 20:22:09 +01:00
static void
2023-05-01 14:24:29 -04:00
teardown ( Fixture * fixture , gconstpointer test_data )
2017-09-21 20:22:09 +01:00
{
/* Recursively remove the temporary directory. */
2023-05-01 14:24:29 -04:00
( void ) glnx_tmpdir_delete ( & fixture - > tmpdir , NULL , NULL ) ;
2017-09-21 20:22:09 +01:00
}
/* Test that the hash values for two #OstreeRepo instances pointing at the same
* repository are equal . We can ’ t test anything else , since hash collisions are
* always a possibility . */
static void
2023-05-01 14:24:29 -04:00
test_repo_hash ( Fixture * fixture , gconstpointer test_data )
2017-09-21 20:22:09 +01:00
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
g_autoptr ( OstreeRepo ) repo1 = ostree_repo_create_at (
fixture - > tmpdir . fd , " . " , OSTREE_REPO_MODE_ARCHIVE , NULL , NULL , & error ) ;
2017-09-21 20:22:09 +01:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo2 = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2017-09-21 20:22:09 +01:00
g_assert_no_error ( error ) ;
g_assert_cmpuint ( ostree_repo_hash ( repo1 ) , = = , ostree_repo_hash ( repo2 ) ) ;
}
/* Test that trying to hash a closed repo results in an assertion failure. */
static void
2023-05-01 14:24:29 -04:00
test_repo_hash_closed ( Fixture * fixture , gconstpointer test_data )
2017-09-21 20:22:09 +01:00
{
if ( g_test_subprocess ( ) )
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GFile ) repo_path = g_file_new_for_path ( fixture - > tmpdir . path ) ;
g_autoptr ( OstreeRepo ) repo = ostree_repo_new ( repo_path ) ;
2017-09-21 20:22:09 +01:00
ostree_repo_hash ( repo ) ;
return ;
}
g_test_trap_subprocess ( NULL , 0 , 0 ) ;
g_test_trap_assert_failed ( ) ;
g_test_trap_assert_stderr ( " *ERROR*ostree_repo_hash: assertion failed:* " ) ;
}
/* Test that various repositories test equal (or not) with each other. */
static void
2023-05-01 14:24:29 -04:00
test_repo_equal ( Fixture * fixture , gconstpointer test_data )
2017-09-21 20:22:09 +01:00
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
2017-09-21 20:22:09 +01:00
/* Create a few separate repos and some #OstreeRepo objects for them. */
glnx_ensure_dir ( fixture - > tmpdir . fd , " repo1 " , 0755 , & error ) ;
g_assert_no_error ( error ) ;
glnx_ensure_dir ( fixture - > tmpdir . fd , " repo2 " , 0755 , & error ) ;
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo1 = ostree_repo_create_at (
fixture - > tmpdir . fd , " repo1 " , OSTREE_REPO_MODE_ARCHIVE , NULL , NULL , & error ) ;
2017-09-21 20:22:09 +01:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo1_alias
= ostree_repo_open_at ( fixture - > tmpdir . fd , " repo1 " , NULL , & error ) ;
2017-09-21 20:22:09 +01:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo2 = ostree_repo_create_at (
fixture - > tmpdir . fd , " repo2 " , OSTREE_REPO_MODE_ARCHIVE , NULL , NULL , & error ) ;
2017-09-21 20:22:09 +01:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( GFile ) closed_repo_path = g_file_new_for_path ( fixture - > tmpdir . path ) ;
g_autoptr ( OstreeRepo ) closed_repo = ostree_repo_new ( closed_repo_path ) ;
2017-09-21 20:22:09 +01:00
/* Test various equalities. */
g_assert_true ( ostree_repo_equal ( repo1 , repo1 ) ) ;
g_assert_true ( ostree_repo_equal ( repo1_alias , repo1_alias ) ) ;
g_assert_true ( ostree_repo_equal ( repo1 , repo1_alias ) ) ;
g_assert_true ( ostree_repo_equal ( repo1_alias , repo1 ) ) ;
g_assert_true ( ostree_repo_equal ( repo2 , repo2 ) ) ;
g_assert_false ( ostree_repo_equal ( repo1 , repo2 ) ) ;
g_assert_false ( ostree_repo_equal ( repo1_alias , repo2 ) ) ;
g_assert_false ( ostree_repo_equal ( repo2 , repo1 ) ) ;
g_assert_false ( ostree_repo_equal ( repo2 , repo1_alias ) ) ;
g_assert_false ( ostree_repo_equal ( repo1 , closed_repo ) ) ;
g_assert_false ( ostree_repo_equal ( repo1_alias , closed_repo ) ) ;
g_assert_false ( ostree_repo_equal ( closed_repo , repo1 ) ) ;
g_assert_false ( ostree_repo_equal ( closed_repo , repo1_alias ) ) ;
g_assert_false ( ostree_repo_equal ( repo2 , closed_repo ) ) ;
g_assert_false ( ostree_repo_equal ( closed_repo , repo2 ) ) ;
g_assert_false ( ostree_repo_equal ( closed_repo , closed_repo ) ) ;
}
2018-08-31 20:20:23 +05:30
static void
2023-05-01 14:24:29 -04:00
test_repo_get_min_free_space ( Fixture * fixture , gconstpointer test_data )
2018-08-31 20:20:23 +05:30
{
g_autoptr ( GKeyFile ) config = NULL ;
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
2018-09-14 20:22:37 +05:30
guint64 bytes = 0 ;
2018-08-31 20:20:23 +05:30
typedef struct
2023-05-01 14:24:29 -04:00
{
const char * val ;
gboolean should_succeed ;
} min_free_space_value ;
2018-08-31 20:20:23 +05:30
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo = ostree_repo_create_at (
fixture - > tmpdir . fd , " . " , OSTREE_REPO_MODE_ARCHIVE , NULL , NULL , & error ) ;
2018-08-31 20:20:23 +05:30
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
min_free_space_value values_to_test [ ]
= { { " 500MB " , TRUE } ,
{ " 0MB " , TRUE } ,
{ " 17179869185GB " , FALSE } , /* Overflow parameter: bytes > G_MAXUINT64 */
{ NULL , FALSE } } ;
2018-08-31 20:20:23 +05:30
config = ostree_repo_copy_config ( repo ) ;
for ( guint i = 0 ; values_to_test [ i ] . val ! = NULL ; i + + )
{
g_key_file_remove_key ( config , " core " , " min-free-space-size " , NULL ) ;
g_key_file_set_string ( config , " core " , " min-free-space-size " , values_to_test [ i ] . val ) ;
ostree_repo_write_config ( repo , config , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_reload_config ( repo , NULL , & error ) ;
g_assert_no_error ( error ) ;
2018-09-14 20:22:37 +05:30
ostree_repo_get_min_free_space_bytes ( repo , & bytes , & error ) ;
2018-08-31 20:20:23 +05:30
if ( values_to_test [ i ] . should_succeed )
g_assert_no_error ( error ) ;
else
continue ;
}
}
2021-04-07 21:03:15 +00:00
static void
2023-05-01 14:24:29 -04:00
test_write_regfile_api ( Fixture * fixture , gconstpointer test_data )
2021-04-07 21:03:15 +00:00
{
g_autoptr ( GKeyFile ) config = NULL ;
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
2021-04-07 21:03:15 +00:00
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo = ostree_repo_create_at (
fixture - > tmpdir . fd , " . " , OSTREE_REPO_MODE_ARCHIVE , NULL , NULL , & error ) ;
2021-04-07 21:03:15 +00:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_auto ( GVariantBuilder ) xattrs_builder ;
g_variant_builder_init ( & xattrs_builder , ( GVariantType * ) " a(ayay) " ) ;
g_variant_builder_add ( & xattrs_builder , " (^ay^ay) " , " security.selinux " ,
" system_u:object_r:etc_t:s0 " ) ;
g_autoptr ( GVariant ) xattrs = g_variant_ref_sink ( g_variant_builder_end ( & xattrs_builder ) ) ;
2021-04-07 21:03:15 +00:00
// Current contents of /etc/networks in Fedora
static const char contents [ ] = " default 0.0.0.0 \n loopback 127.0.0.0 \n link-local 169.254.0.0 \n " ;
// First with no xattrs
2023-05-01 14:24:29 -04:00
g_autofree char * checksum = ostree_repo_write_regfile_inline (
repo , NULL , 0 , 0 , S_IFREG | 0644 , NULL , ( const guint8 * ) contents , sizeof ( contents ) - 1 , NULL ,
& error ) ;
2021-04-07 21:03:15 +00:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_assert_cmpstr ( checksum , = = ,
" 8aaa9dc13a0c5839fe4a277756798c609c53fac6fa2290314ecfef9041065873 " ) ;
2021-04-07 21:03:15 +00:00
g_clear_pointer ( & checksum , g_free ) ;
// Invalid checksum
2023-05-01 14:24:29 -04:00
checksum = ostree_repo_write_regfile_inline (
repo , " 3272139f889f6a7007b3d64adc74be9e2979bf6bbe663d1512e5bd43f4de24a1 " , 0 , 0 ,
S_IFREG | 0644 , NULL , ( const guint8 * ) contents , sizeof ( contents ) - 1 , NULL , & error ) ;
2021-04-07 21:03:15 +00:00
g_assert ( checksum = = NULL ) ;
g_assert ( error ! = NULL ) ;
g_clear_error ( & error ) ;
2023-02-07 14:48:15 -07:00
// Now with xattrs
2021-04-07 21:03:15 +00:00
g_clear_pointer ( & checksum , g_free ) ;
2023-05-01 14:24:29 -04:00
checksum = ostree_repo_write_regfile_inline ( repo , NULL , 0 , 0 , S_IFREG | 0644 , xattrs ,
( const guint8 * ) contents , sizeof ( contents ) - 1 ,
NULL , & error ) ;
2021-04-07 21:03:15 +00:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_assert_cmpstr ( checksum , = = ,
" 4f600d252338f93279c51c964915cb2c26f0d09082164c54890d1a3c78cdeb1e " ) ;
2021-04-07 21:03:15 +00:00
g_clear_pointer ( & checksum , g_free ) ;
2021-04-07 21:03:15 +00:00
// Test symlinks
g_clear_pointer ( & xattrs , g_variant_unref ) ;
2023-05-01 14:24:29 -04:00
g_variant_builder_init ( & xattrs_builder , ( GVariantType * ) " a(ayay) " ) ;
g_variant_builder_add ( & xattrs_builder , " (^ay^ay) " , " security.selinux " ,
" system_u:object_r:bin_t:s0 " ) ;
2021-04-07 21:03:15 +00:00
xattrs = g_variant_ref_sink ( g_variant_builder_end ( & xattrs_builder ) ) ;
g_clear_pointer ( & checksum , g_free ) ;
checksum = ostree_repo_write_symlink ( repo , NULL , 0 , 0 , xattrs , " bash " , NULL , & error ) ;
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_assert_cmpstr ( checksum , = = ,
" 23a2e97d21d960ac7a4e39a8721b1baff7b213e00e5e5641334f50506012fcff " ) ;
2021-04-07 21:03:15 +00:00
}
2021-04-14 20:43:53 -04:00
/* Just a sanity check of the C autolocking API */
static void
2023-05-01 14:24:29 -04:00
test_repo_autolock ( Fixture * fixture , gconstpointer test_data )
2021-04-14 20:43:53 -04:00
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
g_autoptr ( OstreeRepo ) repo = ostree_repo_create_at (
fixture - > tmpdir . fd , " . " , OSTREE_REPO_MODE_ARCHIVE , NULL , NULL , & error ) ;
2021-04-14 20:43:53 -04:00
g_assert_no_error ( error ) ;
{
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepoAutoLock ) lock
= ostree_repo_auto_lock_push ( repo , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
2021-04-14 20:43:53 -04:00
g_assert_no_error ( error ) ;
}
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepoAutoLock ) lock1
= ostree_repo_auto_lock_push ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
2021-04-14 20:43:53 -04:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepoAutoLock ) lock2
= ostree_repo_auto_lock_push ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
2021-04-14 20:43:53 -04:00
g_assert_no_error ( error ) ;
}
2021-04-22 09:10:15 -06:00
/* Locking from single thread with a single OstreeRepo */
static void
2023-05-01 14:24:29 -04:00
test_repo_lock_single ( Fixture * fixture , gconstpointer test_data )
2021-04-22 09:10:15 -06:00
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
g_autoptr ( OstreeRepo ) repo = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
/* Single thread on a single repo can freely recurse in any state */
ostree_repo_lock_push ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_push ( repo , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_push ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
}
/* Unlocking without having ever locked */
static void
2023-05-01 14:24:29 -04:00
test_repo_lock_unlock_never_locked ( Fixture * fixture , gconstpointer test_data )
2021-04-22 09:10:15 -06:00
{
if ( g_test_subprocess ( ) )
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
g_autoptr ( OstreeRepo ) repo = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
return ;
}
g_test_trap_subprocess ( NULL , 0 , 0 ) ;
g_test_trap_assert_failed ( ) ;
g_test_trap_assert_stderr ( " *ERROR*Cannot pop repo never locked repo lock \n " ) ;
}
/* Unlocking after already unlocked */
static void
2023-05-01 14:24:29 -04:00
test_repo_lock_double_unlock ( Fixture * fixture , gconstpointer test_data )
2021-04-22 09:10:15 -06:00
{
if ( g_test_subprocess ( ) )
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
g_autoptr ( OstreeRepo ) repo = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
ostree_repo_lock_push ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
return ;
}
g_test_trap_subprocess ( NULL , 0 , 0 ) ;
g_test_trap_assert_failed ( ) ;
g_test_trap_assert_stderr ( " *ERROR*Cannot pop already unlocked repo lock \n " ) ;
}
/* Unlocking the wrong type */
static void
2023-05-01 14:24:29 -04:00
test_repo_lock_unlock_wrong_type ( Fixture * fixture , gconstpointer test_data )
2021-04-22 09:10:15 -06:00
{
if ( g_test_subprocess ( ) )
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
g_autoptr ( OstreeRepo ) repo = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
ostree_repo_lock_push ( repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
return ;
}
g_test_trap_subprocess ( NULL , 0 , 0 ) ;
g_test_trap_assert_failed ( ) ;
2023-05-01 14:24:29 -04:00
g_test_trap_assert_stderr (
" *ERROR*Repo exclusive lock pop requested, but none have been taken \n " ) ;
2021-04-22 09:10:15 -06:00
}
/* Locking with single thread and multiple OstreeRepos */
static void
2023-05-01 14:24:29 -04:00
test_repo_lock_multi_repo ( Fixture * fixture , gconstpointer test_data )
2021-04-22 09:10:15 -06:00
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
2021-04-22 09:10:15 -06:00
/* Open two OstreeRepo instances */
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo1 = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo2 = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
/* Single thread with multiple OstreeRepo's conflict */
ostree_repo_lock_push ( repo1 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_push ( repo2 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_push ( repo1 , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_error ( error , G_IO_ERROR , G_IO_ERROR_WOULD_BLOCK ) ;
g_clear_error ( & error ) ;
ostree_repo_lock_pop ( repo1 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo2 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
/* Recursive lock should stay exclusive once acquired */
ostree_repo_lock_push ( repo1 , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_push ( repo1 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_push ( repo2 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_error ( error , G_IO_ERROR , G_IO_ERROR_WOULD_BLOCK ) ;
g_clear_error ( & error ) ;
ostree_repo_lock_push ( repo2 , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_error ( error , G_IO_ERROR , G_IO_ERROR_WOULD_BLOCK ) ;
g_clear_error ( & error ) ;
ostree_repo_lock_pop ( repo1 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
ostree_repo_lock_pop ( repo1 , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_no_error ( error ) ;
}
/* Locking from multiple threads with a single OstreeRepo */
2023-05-01 14:24:29 -04:00
typedef struct
{
2021-04-22 09:10:15 -06:00
OstreeRepo * repo ;
guint step ;
} LockThreadData ;
static gpointer
lock_thread1 ( gpointer thread_data )
{
LockThreadData * data = thread_data ;
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
2021-04-22 09:10:15 -06:00
/* Step 0: Take an exclusive lock */
g_assert_cmpuint ( data - > step , = = , 0 ) ;
g_test_message ( " Thread 1: Push exclusive lock " ) ;
ostree_repo_lock_push ( data - > repo , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_no_error ( error ) ;
data - > step + + ;
/* Step 2: Take a shared lock */
while ( data - > step ! = 2 )
g_thread_yield ( ) ;
g_test_message ( " Thread 1: Push shared lock " ) ;
ostree_repo_lock_push ( data - > repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
data - > step + + ;
/* Step 4: Pop both locks */
while ( data - > step ! = 4 )
g_thread_yield ( ) ;
g_test_message ( " Thread 1: Pop shared lock " ) ;
ostree_repo_lock_pop ( data - > repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
g_test_message ( " Thread 1: Pop exclusive lock " ) ;
ostree_repo_lock_pop ( data - > repo , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_no_error ( error ) ;
data - > step + + ;
return NULL ;
}
static gpointer
lock_thread2 ( gpointer thread_data )
{
LockThreadData * data = thread_data ;
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
2021-04-22 09:10:15 -06:00
/* Step 1: Wait for the other thread to acquire a lock and then take a
* shared lock .
*/
while ( data - > step ! = 1 )
g_thread_yield ( ) ;
g_test_message ( " Thread 2: Push shared lock " ) ;
ostree_repo_lock_push ( data - > repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
data - > step + + ;
/* Step 6: Pop lock */
while ( data - > step ! = 6 )
g_thread_yield ( ) ;
g_test_message ( " Thread 2: Pop shared lock " ) ;
ostree_repo_lock_pop ( data - > repo , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
data - > step + + ;
return NULL ;
}
static void
2023-05-01 14:24:29 -04:00
test_repo_lock_multi_thread ( Fixture * fixture , gconstpointer test_data )
2021-04-22 09:10:15 -06:00
{
2023-05-01 14:24:29 -04:00
g_autoptr ( GError ) error = NULL ;
g_autoptr ( OstreeRepo ) repo1 = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
g_autoptr ( OstreeRepo ) repo2 = ostree_repo_open_at ( fixture - > tmpdir . fd , " . " , NULL , & error ) ;
2021-04-22 09:10:15 -06:00
g_assert_no_error ( error ) ;
2023-05-01 14:24:29 -04:00
LockThreadData thread_data = { repo1 , 0 } ;
2021-04-22 09:10:15 -06:00
GThread * thread1 = g_thread_new ( " lock-thread-1 " , lock_thread1 , & thread_data ) ;
GThread * thread2 = g_thread_new ( " lock-thread-2 " , lock_thread2 , & thread_data ) ;
/* Step 3: Try to take a shared lock on repo2. This should fail since
* thread1 still has an exclusive lock .
*/
while ( thread_data . step ! = 3 )
g_thread_yield ( ) ;
g_test_message ( " Repo 2: Push failing shared lock " ) ;
ostree_repo_lock_push ( repo2 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_error ( error , G_IO_ERROR , G_IO_ERROR_WOULD_BLOCK ) ;
g_clear_error ( & error ) ;
thread_data . step + + ;
/* Step 5: Try to a lock on repo2. A shared lock should succeed since
* thread1 has dropped its exclusive lock .
*/
while ( thread_data . step ! = 5 )
g_thread_yield ( ) ;
g_test_message ( " Repo 2: Push shared lock " ) ;
ostree_repo_lock_push ( repo2 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
g_test_message ( " Repo 2: Push failing exclusive lock " ) ;
ostree_repo_lock_push ( repo2 , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_error ( error , G_IO_ERROR , G_IO_ERROR_WOULD_BLOCK ) ;
g_clear_error ( & error ) ;
thread_data . step + + ;
/* Step 7: Now both threads have dropped their locks and taking an exclusive
* lock should succeed .
*/
while ( thread_data . step ! = 7 )
g_thread_yield ( ) ;
g_test_message ( " Repo 2: Push exclusive lock " ) ;
ostree_repo_lock_push ( repo2 , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_no_error ( error ) ;
g_test_message ( " Repo 2: Pop exclusive lock " ) ;
ostree_repo_lock_pop ( repo2 , OSTREE_REPO_LOCK_EXCLUSIVE , NULL , & error ) ;
g_assert_no_error ( error ) ;
g_test_message ( " Repo 2: Pop shared lock " ) ;
ostree_repo_lock_pop ( repo2 , OSTREE_REPO_LOCK_SHARED , NULL , & error ) ;
g_assert_no_error ( error ) ;
thread_data . step + + ;
g_thread_join ( thread1 ) ;
g_thread_join ( thread2 ) ;
}
2017-09-21 20:22:09 +01:00
int
2023-05-01 14:24:29 -04:00
main ( int argc , char * * argv )
2017-09-21 20:22:09 +01:00
{
setlocale ( LC_ALL , " " ) ;
g_test_init ( & argc , & argv , NULL ) ;
2023-05-01 14:24:29 -04:00
g_test_add ( " /repo/hash " , Fixture , NULL , setup , test_repo_hash , teardown ) ;
g_test_add ( " /repo/hash/closed " , Fixture , NULL , setup , test_repo_hash_closed , teardown ) ;
g_test_add ( " /repo/equal " , Fixture , NULL , setup , test_repo_equal , teardown ) ;
g_test_add ( " /repo/get_min_free_space " , Fixture , NULL , setup , test_repo_get_min_free_space ,
teardown ) ;
g_test_add ( " /repo/write_regfile_api " , Fixture , NULL , setup , test_write_regfile_api , teardown ) ;
g_test_add ( " /repo/autolock " , Fixture , NULL , setup , test_repo_autolock , teardown ) ;
g_test_add ( " /repo/lock/single " , Fixture , NULL , lock_setup , test_repo_lock_single , teardown ) ;
2021-04-22 09:10:15 -06:00
g_test_add ( " /repo/lock/unlock-never-locked " , Fixture , NULL , lock_setup ,
test_repo_lock_unlock_never_locked , teardown ) ;
2023-05-01 14:24:29 -04:00
g_test_add ( " /repo/lock/double-unlock " , Fixture , NULL , lock_setup , test_repo_lock_double_unlock ,
teardown ) ;
2021-04-22 09:10:15 -06:00
g_test_add ( " /repo/lock/unlock-wrong-type " , Fixture , NULL , lock_setup ,
test_repo_lock_unlock_wrong_type , teardown ) ;
2023-05-01 14:24:29 -04:00
g_test_add ( " /repo/lock/multi-repo " , Fixture , NULL , lock_setup , test_repo_lock_multi_repo ,
teardown ) ;
g_test_add ( " /repo/lock/multi-thread " , Fixture , NULL , lock_setup , test_repo_lock_multi_thread ,
teardown ) ;
2017-09-21 20:22:09 +01:00
return g_test_run ( ) ;
}