mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-23 12:41:50 +01:00
32b0dd24e3
Some of GLib's unit tests are under an apparently GLib-specific permissive license, vaguely similar to the BSD/MIT family but with the GPL's lack-of-warranty wording. This is not on SPDX's list of well-known licenses, so we need to use a custom license name prefixed with LicenseRef if we want to represent this in SPDX/REUSE syntax. Most of the newer tests seem to be licensed under LGPL-2.1-or-later instead. Signed-off-by: Simon McVittie <smcv@collabora.com>
474 lines
13 KiB
C
474 lines
13 KiB
C
/* GLib testing framework examples and tests
|
|
* Copyright (C) 2019 Руслан Ижбулатов <lrn1986@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: LicenseRef-old-glib-tests
|
|
*
|
|
* This work is provided "as is"; redistribution and modification
|
|
* in whole or in part, in any medium, physical or electronic is
|
|
* permitted without restriction.
|
|
*
|
|
* This work 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.
|
|
*
|
|
* In no event shall the authors or contributors be liable for any
|
|
* direct, indirect, incidental, special, exemplary, or consequential
|
|
* damages (including, but not limited to, procurement of substitute
|
|
* goods or services; loss of use, data, or profits; or business
|
|
* interruption) however caused and on any theory of liability, whether
|
|
* in contract, strict liability, or tort (including negligence or
|
|
* otherwise) arising in any way out of the use of this software, even
|
|
* if advised of the possibility of such damage.
|
|
*/
|
|
|
|
#include <glib/glib.h>
|
|
#include <gio/gio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "../giowin32-private.c"
|
|
|
|
static int
|
|
g_utf16_cmp0 (const gunichar2 *str1,
|
|
const gunichar2 *str2)
|
|
{
|
|
if (!str1)
|
|
return -(str1 != str2);
|
|
if (!str2)
|
|
return str1 != str2;
|
|
|
|
while (TRUE)
|
|
{
|
|
if (str1[0] > str2[0])
|
|
return 1;
|
|
else if (str1[0] < str2[0])
|
|
return -1;
|
|
else if (str1[0] == 0 && str2[0] == 0)
|
|
return 0;
|
|
|
|
str1++;
|
|
str2++;
|
|
}
|
|
}
|
|
|
|
#define g_assert_cmputf16(s1, cmp, s2, s1u8, s2u8) \
|
|
G_STMT_START { \
|
|
const gunichar2 *__s1 = (s1), *__s2 = (s2); \
|
|
if (g_utf16_cmp0 (__s1, __s2) cmp 0) ; else \
|
|
g_assertion_message_cmpstr (G_LOG_DOMAIN, __FILE__, __LINE__, G_STRFUNC, \
|
|
#s1u8 " " #cmp " " #s2u8, s1u8, #cmp, s2u8); \
|
|
} G_STMT_END
|
|
|
|
static void
|
|
test_utf16_strfuncs (void)
|
|
{
|
|
gsize i;
|
|
|
|
struct {
|
|
gsize len;
|
|
const gunichar2 utf16[10];
|
|
const gchar *utf8;
|
|
const gchar *utf8_folded;
|
|
} string_cases[] = {
|
|
{
|
|
0,
|
|
{ 0x0000 },
|
|
"",
|
|
"",
|
|
},
|
|
{
|
|
1,
|
|
{ 0x0020, 0x0000 },
|
|
" ",
|
|
" ",
|
|
},
|
|
{
|
|
2,
|
|
{ 0x0020, 0xd800, 0x0000 },
|
|
NULL,
|
|
NULL,
|
|
},
|
|
};
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (string_cases); i++)
|
|
{
|
|
gsize len;
|
|
gunichar2 *str;
|
|
gboolean success;
|
|
gchar *utf8;
|
|
gchar *utf8_folded;
|
|
|
|
len = g_utf16_len (string_cases[i].utf16);
|
|
g_assert_cmpuint (len, ==, string_cases[i].len);
|
|
|
|
str = (gunichar2 *) g_utf16_find_basename (string_cases[i].utf16, -1);
|
|
/* This only works because all testcases lack separators */
|
|
g_assert_true (string_cases[i].utf16 == str);
|
|
|
|
str = g_wcsdup (string_cases[i].utf16, string_cases[i].len);
|
|
g_assert_cmpmem (string_cases[i].utf16, len, str, len);
|
|
g_free (str);
|
|
|
|
str = g_wcsdup (string_cases[i].utf16, -1);
|
|
g_assert_cmpmem (string_cases[i].utf16, len, str, len);
|
|
g_free (str);
|
|
|
|
success = g_utf16_to_utf8_and_fold (string_cases[i].utf16, -1, NULL, NULL);
|
|
|
|
if (string_cases[i].utf8 == NULL)
|
|
g_assert_false (success);
|
|
else
|
|
g_assert_true (success);
|
|
|
|
utf8 = NULL;
|
|
success = g_utf16_to_utf8_and_fold (string_cases[i].utf16, -1, &utf8, NULL);
|
|
|
|
if (string_cases[i].utf8 != NULL)
|
|
{
|
|
g_assert_true (success);
|
|
g_assert_cmpstr (string_cases[i].utf8, ==, utf8);
|
|
/* This only works because all testcases lack separators */
|
|
g_assert_true (utf8 == g_utf8_find_basename (utf8, len));
|
|
}
|
|
|
|
g_free (utf8);
|
|
|
|
utf8 = NULL;
|
|
utf8_folded = NULL;
|
|
success = g_utf16_to_utf8_and_fold (string_cases[i].utf16, -1, &utf8, &utf8_folded);
|
|
|
|
if (string_cases[i].utf8 != NULL)
|
|
{
|
|
g_assert_true (success);
|
|
g_assert_cmpstr (string_cases[i].utf8_folded, ==, utf8_folded);
|
|
}
|
|
|
|
g_free (utf8);
|
|
g_free (utf8_folded);
|
|
}
|
|
}
|
|
|
|
struct {
|
|
const char *orig;
|
|
const char *executable;
|
|
const char *executable_basename;
|
|
gboolean is_rundll32;
|
|
const char *fixed;
|
|
} rundll32_commandlines[] = {
|
|
{
|
|
"%SystemRoot%\\System32\\rundll32.exe \"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll\", ImageView_Fullscreen %1",
|
|
"%SystemRoot%\\System32\\rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"%SystemRoot%\\System32\\rundll32.exe \"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll\" ImageView_Fullscreen %1",
|
|
},
|
|
{
|
|
"%SystemRoot%/System32/rundll32.exe \"%ProgramFiles%/Windows Photo Viewer/PhotoViewer.dll\", ImageView_Fullscreen %1",
|
|
"%SystemRoot%/System32/rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"%SystemRoot%/System32/rundll32.exe \"%ProgramFiles%/Windows Photo Viewer/PhotoViewer.dll\" ImageView_Fullscreen %1",
|
|
},
|
|
{
|
|
"%SystemRoot%\\System32/rundll32.exe \"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll\", ImageView_Fullscreen %1",
|
|
"%SystemRoot%\\System32/rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"%SystemRoot%\\System32/rundll32.exe \"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll\" ImageView_Fullscreen %1",
|
|
},
|
|
{
|
|
"\"some path with spaces\\rundll32.exe\" \"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll\", ImageView_Fullscreen %1",
|
|
"some path with spaces\\rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"\"some path with spaces\\rundll32.exe\" \"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll\" ImageView_Fullscreen %1",
|
|
},
|
|
{
|
|
" \"some path with spaces\\rundll32.exe\"\"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll\",ImageView_Fullscreen %1",
|
|
"some path with spaces\\rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
" \"some path with spaces\\rundll32.exe\"\"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll\" ImageView_Fullscreen %1",
|
|
},
|
|
{
|
|
"rundll32.exe foo.bar,baz",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"rundll32.exe foo.bar baz",
|
|
},
|
|
{
|
|
" rundll32.exe foo.bar,baz",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
" rundll32.exe foo.bar baz",
|
|
},
|
|
{
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
FALSE,
|
|
NULL,
|
|
},
|
|
{
|
|
"rundll32.exe ,foobar",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
FALSE,
|
|
NULL,
|
|
},
|
|
{
|
|
"rundll32.exe ,foobar",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
FALSE,
|
|
NULL,
|
|
},
|
|
{
|
|
"rundll32.exe foo.dll",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
FALSE,
|
|
NULL,
|
|
},
|
|
{
|
|
"rundll32.exe \"foo bar\",baz",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"rundll32.exe \"foo bar\" baz",
|
|
},
|
|
{
|
|
"\"rundll32.exe\" \"foo bar\",baz",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"\"rundll32.exe\" \"foo bar\" baz",
|
|
},
|
|
{
|
|
"\"rundll32.exe\" \"foo bar\",, , ,,, , ,,baz",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"\"rundll32.exe\" \"foo bar\" , , ,,, , ,,baz",
|
|
},
|
|
{
|
|
"\"rundll32.exe\" foo.bar,,,,,,,,,baz",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"\"rundll32.exe\" foo.bar ,,,,,,,,baz",
|
|
},
|
|
{
|
|
"\"rundll32.exe\" foo.bar baz",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"\"rundll32.exe\" foo.bar baz",
|
|
},
|
|
{
|
|
"\"RuNdlL32.exe\" foo.bar baz",
|
|
"RuNdlL32.exe",
|
|
"RuNdlL32.exe",
|
|
TRUE,
|
|
"\"RuNdlL32.exe\" foo.bar baz",
|
|
},
|
|
{
|
|
"%SystemRoot%\\System32\\rundll32.exe \"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll,\" ImageView_Fullscreen %1",
|
|
"%SystemRoot%\\System32\\rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"%SystemRoot%\\System32\\rundll32.exe \"%ProgramFiles%\\Windows Photo Viewer\\PhotoViewer.dll,\" ImageView_Fullscreen %1",
|
|
},
|
|
{
|
|
"\"rundll32.exe\" \"foo bar,\"baz",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"\"rundll32.exe\" \"foo bar,\"baz",
|
|
},
|
|
{
|
|
"\"rundll32.exe\" some,thing",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"\"rundll32.exe\" some thing",
|
|
},
|
|
{
|
|
"\"rundll32.exe\" some,",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
FALSE,
|
|
"\"rundll32.exe\" some,",
|
|
},
|
|
/* These filenames are not allowed on Windows, but our function doesn't care about that */
|
|
{
|
|
"run\"dll32.exe foo\".bar,baz",
|
|
"run\"dll32.exe",
|
|
"run\"dll32.exe",
|
|
FALSE,
|
|
NULL,
|
|
},
|
|
{
|
|
"run,dll32.exe foo.bar,baz",
|
|
"run,dll32.exe",
|
|
"run,dll32.exe",
|
|
FALSE,
|
|
NULL,
|
|
},
|
|
{
|
|
"\"rundll32.exe\" some, thing",
|
|
"rundll32.exe",
|
|
"rundll32.exe",
|
|
TRUE,
|
|
"\"rundll32.exe\" some thing",
|
|
},
|
|
/* Commands with "rundll32" (without the .exe suffix) do exist,
|
|
* but GLib currently does not recognize them, so there's no point
|
|
* in testing these.
|
|
*/
|
|
};
|
|
|
|
static void
|
|
test_win32_rundll32_fixup (void)
|
|
{
|
|
gsize i;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (rundll32_commandlines); i++)
|
|
{
|
|
gunichar2 *argument;
|
|
gunichar2 *expected;
|
|
|
|
if (!rundll32_commandlines[i].is_rundll32)
|
|
continue;
|
|
|
|
argument = g_utf8_to_utf16 (rundll32_commandlines[i].orig, -1, NULL, NULL, NULL);
|
|
expected = g_utf8_to_utf16 (rundll32_commandlines[i].fixed, -1, NULL, NULL, NULL);
|
|
|
|
g_assert_nonnull (argument);
|
|
g_assert_nonnull (expected);
|
|
_g_win32_fixup_broken_microsoft_rundll_commandline (argument);
|
|
|
|
g_assert_cmputf16 (argument, ==, expected, rundll32_commandlines[i].orig, rundll32_commandlines[i].fixed);
|
|
|
|
g_free (argument);
|
|
g_free (expected);
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_win32_extract_executable (void)
|
|
{
|
|
gsize i;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (rundll32_commandlines); i++)
|
|
{
|
|
gunichar2 *argument;
|
|
gchar *dll_function;
|
|
gchar *executable;
|
|
gchar *executable_basename;
|
|
gchar *executable_folded;
|
|
gchar *executable_folded_basename;
|
|
|
|
argument = g_utf8_to_utf16 (rundll32_commandlines[i].orig, -1, NULL, NULL, NULL);
|
|
|
|
_g_win32_extract_executable (argument, NULL, NULL, NULL, NULL, &dll_function);
|
|
|
|
if (rundll32_commandlines[i].is_rundll32)
|
|
g_assert_nonnull (dll_function);
|
|
else
|
|
g_assert_null (dll_function);
|
|
|
|
g_free (dll_function);
|
|
|
|
executable = NULL;
|
|
executable_basename = NULL;
|
|
executable_folded = NULL;
|
|
executable_folded_basename = NULL;
|
|
_g_win32_extract_executable (argument, &executable, &executable_basename, &executable_folded, &executable_folded_basename, NULL);
|
|
|
|
g_assert_cmpstr (rundll32_commandlines[i].executable, ==, executable);
|
|
g_assert_cmpstr (rundll32_commandlines[i].executable_basename, ==, executable_basename);
|
|
g_assert_nonnull (executable_folded);
|
|
|
|
g_free (executable);
|
|
g_free (executable_folded);
|
|
|
|
/* Check the corner-case where we don't want to know where basename is */
|
|
executable = NULL;
|
|
executable_folded = NULL;
|
|
_g_win32_extract_executable (argument, &executable, NULL, &executable_folded, NULL, NULL);
|
|
|
|
g_assert_cmpstr (rundll32_commandlines[i].executable, ==, executable);
|
|
g_assert_nonnull (executable_folded);
|
|
|
|
g_free (executable);
|
|
g_free (executable_folded);
|
|
|
|
g_free (argument);
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_win32_parse_filename (void)
|
|
{
|
|
gsize i;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (rundll32_commandlines); i++)
|
|
{
|
|
gunichar2 *argument;
|
|
argument = g_utf8_to_utf16 (rundll32_commandlines[i].orig, -1, NULL, NULL, NULL);
|
|
/* Just checking that it doesn't blow up on various (sometimes incorrect) strings */
|
|
_g_win32_parse_filename (argument, FALSE, NULL, NULL, NULL, NULL);
|
|
g_free (argument);
|
|
}
|
|
}
|
|
|
|
static void
|
|
do_fail_on_broken_utf16_1 (void)
|
|
{
|
|
const gunichar2 utf16[] = { 0xd800, 0x0000 };
|
|
_g_win32_extract_executable (utf16, NULL, NULL, NULL, NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
do_fail_on_broken_utf16_2 (void)
|
|
{
|
|
/* "rundll32.exe <invalid utf16> r" */
|
|
gchar *dll_function;
|
|
const gunichar2 utf16[] = { 0x0072, 0x0075, 0x006E, 0x0064, 0x006C, 0x006C, 0x0033, 0x0032,
|
|
0x002E, 0x0065, 0x0078, 0x0065, 0x0020, 0xd800, 0x0020, 0x0072, 0x0000 };
|
|
_g_win32_extract_executable (utf16, NULL, NULL, NULL, NULL, &dll_function);
|
|
}
|
|
|
|
static void
|
|
test_fail_on_broken_utf16 (void)
|
|
{
|
|
g_test_trap_subprocess ("/appinfo/subprocess/win32-assert-broken-utf16_1", 0,
|
|
G_TEST_SUBPROCESS_DEFAULT);
|
|
g_test_trap_assert_failed ();
|
|
g_test_trap_assert_stderr ("*GLib-GIO:ERROR:*giowin32-private.c:*:_g_win32_extract_executable: assertion failed: (folded)*");
|
|
g_test_trap_subprocess ("/appinfo/subprocess/win32-assert-broken-utf16_2", 0,
|
|
G_TEST_SUBPROCESS_DEFAULT);
|
|
g_test_trap_assert_failed ();
|
|
g_test_trap_assert_stderr ("*GLib-GIO:ERROR:*giowin32-private.c:*:_g_win32_extract_executable: assertion failed: (folded)*");
|
|
}
|
|
|
|
int
|
|
main (int argc,
|
|
char *argv[])
|
|
{
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
g_test_add_func ("/appinfo/utf16-strfuncs", test_utf16_strfuncs);
|
|
g_test_add_func ("/appinfo/win32-extract-executable", test_win32_extract_executable);
|
|
g_test_add_func ("/appinfo/win32-rundll32-fixup", test_win32_rundll32_fixup);
|
|
g_test_add_func ("/appinfo/win32-parse-filename", test_win32_parse_filename);
|
|
g_test_add_func ("/appinfo/win32-utf16-conversion-fail", test_fail_on_broken_utf16);
|
|
|
|
g_test_add_func ("/appinfo/subprocess/win32-assert-broken-utf16_1", do_fail_on_broken_utf16_1);
|
|
g_test_add_func ("/appinfo/subprocess/win32-assert-broken-utf16_2", do_fail_on_broken_utf16_2);
|
|
|
|
return g_test_run ();
|
|
}
|