| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366 |
- /*
- * libcsync -- a library to sync a directory with another
- *
- * Copyright (c) 2015-2013 by Klaas Freitag <freitag@owncloud.com>
- *
- * 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.1 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
- * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- */
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <fcntl.h>
- #include <cstring>
- #include <cerrno>
- #include <cstdio>
- #include "csync.h"
- #include "vio/csync_vio_local.h"
- #include <QDir>
- static const auto CSYNC_TEST_DIR = []{ return QStringLiteral("%1/csync_test").arg(QDir::tempPath());}();
- #include "torture.h"
- namespace {
- int oc_mkdir(const QString &path)
- {
- return QDir(path).mkpath(path) ? 0 : -1;
- }
- }
- #define WD_BUFFER_SIZE 255
- static mbchar_t wd_buffer[WD_BUFFER_SIZE];
- using statevar = struct {
- QByteArray result;
- QByteArray ignored_dir;
- };
- /* remove the complete test dir */
- static int wipe_testdir()
- {
- QDir tmp(CSYNC_TEST_DIR);
- if (tmp.exists())
- {
- return tmp.removeRecursively() ? 0 : 1;
- }
- return 0;
- }
- static int setup_testenv(void **state) {
- int rc = 0;
- rc = wipe_testdir();
- assert_int_equal(rc, 0);
- auto dir = CSYNC_TEST_DIR;
- rc = oc_mkdir(dir);
- assert_int_equal(rc, 0);
- assert_non_null(_tgetcwd(wd_buffer, WD_BUFFER_SIZE));
- #ifdef Q_OS_WIN
- rc = _tchdir(dir.toStdWString().data());
- #else
- rc = _tchdir(dir.toLocal8Bit().constData());
- #endif
- assert_int_equal(rc, 0);
- /* --- initialize csync */
- auto mystate = new statevar;
- *state = mystate;
- return 0;
- }
- static void output( const char *text )
- {
- printf("%s\n", text);
- }
- static int teardown(void **state) {
- int rc = -1;
- output("================== Tearing down!\n");
- rc = _tchdir(wd_buffer);
- assert_int_equal(rc, 0);
- rc = wipe_testdir();
- assert_int_equal(rc, 0);
- delete reinterpret_cast<statevar*>(*state);
- return 0;
- }
- /* This function takes a relative path, prepends it with the CSYNC_TEST_DIR
- * and creates each sub directory.
- */
- static void create_dirs( const char *path )
- {
- int rc = -1;
- auto _mypath = QStringLiteral("%1/%2").arg(CSYNC_TEST_DIR, QString::fromUtf8(path)).toUtf8();
- char *mypath = _mypath.data();
- char *p = mypath + CSYNC_TEST_DIR.size() + 1; /* start behind the offset */
- int i = 0;
- assert_non_null(path);
- while( *(p+i) ) {
- if( *(p+i) == '/' ) {
- p[i] = '\0';
- auto mb_dir = QString::fromUtf8(mypath);
- rc = oc_mkdir(mb_dir);
- if(rc)
- {
- rc = errno;
- }
- assert_int_equal(rc, 0);
- p[i] = '/';
- }
- i++;
- }
- }
- /*
- * This function uses the vio_opendir, vio_readdir and vio_closedir functions
- * to traverse a file tree that was created before by the create_dir function.
- *
- * It appends a listing to the result member of the incoming struct in *state
- * that can be compared later to what was expected in the calling functions.
- *
- * The int parameter cnt contains the number of seen files (not dirs) in the
- * whole tree.
- *
- */
- static void traverse_dir(void **state, const QString &dir, int *cnt)
- {
- csync_vio_handle_t *dh = nullptr;
- std::unique_ptr<csync_file_stat_t> dirent;
- auto sv = (statevar*) *state;
- QByteArray subdir;
- QByteArray subdir_out;
- int rc = -1;
- int is_dir = 0;
- dh = csync_vio_local_opendir(dir);
- assert_non_null(dh);
- OCC::Vfs *vfs = nullptr;
- while( (dirent = csync_vio_local_readdir(dh, vfs)) ) {
- assert_non_null(dirent.get());
- if (!dirent->original_path.isEmpty()) {
- sv->ignored_dir = dirent->original_path;
- continue;
- }
- assert_false(dirent->path.isEmpty());
- if( dirent->path == ".." || dirent->path == "." ) {
- continue;
- }
- is_dir = (dirent->type == ItemTypeDirectory) ? 1:0;
- subdir = dir.toUtf8() + "/" + dirent->path;
- subdir_out = (is_dir ? "<DIR> ":" ") + subdir;
- if( is_dir ) {
- if( sv->result.isNull() ) {
- sv->result = subdir_out;
- } else {
- sv->result += subdir_out;
- }
- } else {
- *cnt = *cnt +1;
- }
- output(subdir_out.constData());
- if( is_dir ) {
- traverse_dir(state, QString::fromUtf8(subdir), cnt);
- }
- }
- rc = csync_vio_local_closedir(dh);
- assert_int_equal(rc, 0);
- }
- static void create_file( const char *path, const char *name, const char *content)
- {
- QFile file(QStringLiteral("%1/%2%3").arg(CSYNC_TEST_DIR, QString::fromUtf8(path), QString::fromUtf8(name)));
- assert_int_equal(1, file.open(QIODevice::WriteOnly | QIODevice::NewOnly));
- file.write(content);
- }
- static void check_readdir_shorttree(void **state)
- {
- auto sv = (statevar*) *state;
- const char *t1 = "alibaba/und/die/vierzig/räuber/";
- create_dirs( t1 );
- int files_cnt = 0;
-
- traverse_dir(state, CSYNC_TEST_DIR, &files_cnt);
- assert_string_equal(sv->result.constData(),
- QString::fromUtf8("<DIR> %1/alibaba"
- "<DIR> %1/alibaba/und"
- "<DIR> %1/alibaba/und/die"
- "<DIR> %1/alibaba/und/die/vierzig"
- "<DIR> %1/alibaba/und/die/vierzig/räuber")
- .arg(CSYNC_TEST_DIR)
- .toUtf8()
- .constData());
- assert_int_equal(files_cnt, 0);
- }
- static void check_readdir_with_content(void **state)
- {
- auto sv = (statevar*) *state;
- int files_cnt = 0;
- const char *t1 = "warum/nur/40/Räuber/";
- create_dirs( t1 );
- create_file( t1, "Räuber Max.txt", "Der Max ist ein schlimmer finger");
- create_file( t1, "пя́тница.txt", "Am Freitag tanzt der Ürk");
- traverse_dir(state, CSYNC_TEST_DIR, &files_cnt);
- assert_string_equal(sv->result.constData(),
- QString::fromUtf8("<DIR> %1/warum"
- "<DIR> %1/warum/nur"
- "<DIR> %1/warum/nur/40"
- "<DIR> %1/warum/nur/40/Räuber")
- .arg(CSYNC_TEST_DIR)
- .toUtf8()
- .constData());
- /* " %1/warum/nur/40/Räuber/Räuber Max.txt"
- " %1/warum/nur/40/Räuber/пя́тница.txt"; */
- assert_int_equal(files_cnt, 2); /* Two files in the sub dir */
- }
- static void check_readdir_longtree(void **state)
- {
- auto sv = (statevar*) *state;
- /* Strange things here: Compilers only support strings with length of 4k max.
- * The expected result string is longer, so it needs to be split up in r1, r2 and r3
- */
- /* create the test tree */
- const char *t1 = "vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns/KISTE/ooooooooooooooooooooooooooohhhhhh/und/BESSER/ZWEI/Butteln/VOLL RUM/";
- create_dirs( t1 );
- const auto r1 = QString::fromUtf8(
- "<DIR> %1/vierzig"
- "<DIR> %1/vierzig/mann"
- "<DIR> %1/vierzig/mann/auf"
- "<DIR> %1/vierzig/mann/auf/des"
- "<DIR> %1/vierzig/mann/auf/des/toten"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum").arg(CSYNC_TEST_DIR);
- const auto r2 = QString::fromUtf8(
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH").arg(CSYNC_TEST_DIR);
- const auto r3 = QString::fromUtf8(
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns/KISTE"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns/KISTE/ooooooooooooooooooooooooooohhhhhh"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns/KISTE/ooooooooooooooooooooooooooohhhhhh/und"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns/KISTE/ooooooooooooooooooooooooooohhhhhh/und/BESSER"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns/KISTE/ooooooooooooooooooooooooooohhhhhh/und/BESSER/ZWEI"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns/KISTE/ooooooooooooooooooooooooooohhhhhh/und/BESSER/ZWEI/Butteln"
- "<DIR> %1/vierzig/mann/auf/des/toten/Mann/kiste/ooooooooooooooooooooooh/and/ne/bottle/voll/rum/und/so/singen/wir/VIERZIG/MANN/AUF/DES/TOTEN/MANNS/KISTE/OOOOOOOOH/AND/NE/BOTTLE/VOLL/RUM/undnochmalallezusammen/VierZig/MannaufDesTotenManns/KISTE/ooooooooooooooooooooooooooohhhhhh/und/BESSER/ZWEI/Butteln/VOLL RUM").arg(CSYNC_TEST_DIR);
- /* assemble the result string ... */
- const auto result = (r1 + r2 + r3).toUtf8();
- int files_cnt = 0;
- traverse_dir(state, CSYNC_TEST_DIR, &files_cnt);
- assert_int_equal(files_cnt, 0);
- /* and compare. */
- assert_string_equal(sv->result.constData(), result.constData());
- }
- // https://github.com/owncloud/client/issues/3128 https://github.com/owncloud/client/issues/2777
- static void check_readdir_bigunicode(void **state)
- {
- auto sv = (statevar*) *state;
- // 1: ? ASCII: 239 - EF
- // 2: ? ASCII: 187 - BB
- // 3: ? ASCII: 191 - BF
- // 4: ASCII: 32 - 20
- QString p = QStringLiteral("%1/%2").arg(CSYNC_TEST_DIR, QStringLiteral("goodone/"));
- int rc = oc_mkdir(p);
- assert_int_equal(rc, 0);
- p = QStringLiteral("%1/goodone/ugly\xEF\xBB\xBF\x32.txt").arg(CSYNC_TEST_DIR); // file with encoding error
- rc = oc_mkdir(p);
- assert_int_equal(rc, 0);
- int files_cnt = 0;
- traverse_dir(state, CSYNC_TEST_DIR, &files_cnt);
- const auto expected_result = QStringLiteral("<DIR> %1/goodone"
- "<DIR> %1/goodone/ugly\xEF\xBB\xBF\x32.txt")
- .arg(CSYNC_TEST_DIR);
- assert_string_equal(sv->result.constData(), expected_result.toUtf8().constData());
- assert_int_equal(files_cnt, 0);
- }
- int torture_run_tests(void)
- {
- const struct CMUnitTest tests[] = {
- cmocka_unit_test_setup_teardown(check_readdir_shorttree, setup_testenv, teardown),
- cmocka_unit_test_setup_teardown(check_readdir_with_content, setup_testenv, teardown),
- cmocka_unit_test_setup_teardown(check_readdir_longtree, setup_testenv, teardown),
- cmocka_unit_test_setup_teardown(check_readdir_bigunicode, setup_testenv, teardown),
- };
- return cmocka_run_group_tests(tests, nullptr, nullptr);
- }
|