/* * Copyright (c) 2018 Stefan Sperling * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "got_error.h" #include "got_object.h" #include "got_reference.h" #include "got_repository.h" #include "got_worktree.h" #include "got_opentemp.h" #include "got_lib_worktree.h" #include "got_lib_path.h" #define GOT_REPO_PATH "../../../" static int verbose; void test_printf(char *fmt, ...) { va_list ap; if (!verbose) return; va_start(ap, fmt); vprintf(fmt, ap); va_end(ap); } static int remove_got_dir(const char *worktree_path) { char *path; if (asprintf(&path, "%s/%s", worktree_path, GOT_WORKTREE_GOT_DIR) == -1) return 0; rmdir(path); free(path); return 1; } static int remove_meta_file(const char *worktree_path, const char *name) { char *path; if (asprintf(&path, "%s/%s/%s", worktree_path, GOT_WORKTREE_GOT_DIR, name) == -1) return 0; unlink(path); free(path); return 1; } static int remove_worktree(const char *worktree_path) { if (!remove_meta_file(worktree_path, GOT_WORKTREE_HEAD)) return 0; if (!remove_meta_file(worktree_path, GOT_WORKTREE_FILE_INDEX)) return 0; if (!remove_meta_file(worktree_path, GOT_WORKTREE_REPOSITORY)) return 0; if (!remove_meta_file(worktree_path, GOT_WORKTREE_PATH_PREFIX)) return 0; if (!remove_meta_file(worktree_path, GOT_WORKTREE_LOCK)) return 0; if (!remove_meta_file(worktree_path, GOT_WORKTREE_FORMAT)) return 0; if (!remove_got_dir(worktree_path)) return 0; if (rmdir(worktree_path) == -1) return 0; return 1; } static int read_meta_file(char **content, const char *path) { FILE *f; size_t len; const char delim[3] = {'\0', '\0', '\0'}; int ret = 0; f = fopen(path, "r"); if (f == NULL) return errno; *content = fparseln(f, &len, NULL, delim, 0); if (*content == NULL) ret = errno; fclose(f); return ret; } static int check_meta_file_exists(const char *worktree_path, const char *name) { struct stat sb; char *path; int ret = 0; if (asprintf(&path, "%s/%s/%s", worktree_path, GOT_WORKTREE_GOT_DIR, name) == -1) return 0; if (stat(path, &sb) == 0) ret = 1; if (verbose) { char *content; if (read_meta_file(&content, path) == 0) { test_printf("%s:\t%s\n", name, content); free(content); } } free(path); return ret; } static int worktree_init(const char *repo_path) { const struct got_error *err; struct got_repository *repo = NULL; struct got_reference *head_ref = NULL; char worktree_path[PATH_MAX]; int ok = 0; err = got_repo_open(&repo, repo_path); if (err != NULL || repo == NULL) goto done; err = got_ref_open(&head_ref, repo, GOT_REF_HEAD); if (err != NULL || head_ref == NULL) goto done; strlcpy(worktree_path, "worktree-XXXXXX", sizeof(worktree_path)); if (mkdtemp(worktree_path) == NULL) goto done; err = got_worktree_init(worktree_path, head_ref, "/", repo); if (err != NULL) goto done; /* Ensure required files were created. */ if (!check_meta_file_exists(worktree_path, GOT_WORKTREE_HEAD)) goto done; if (!check_meta_file_exists(worktree_path, GOT_WORKTREE_LOCK)) goto done; if (!check_meta_file_exists(worktree_path, GOT_WORKTREE_FILE_INDEX)) goto done; if (!check_meta_file_exists(worktree_path, GOT_WORKTREE_REPOSITORY)) goto done; if (!check_meta_file_exists(worktree_path, GOT_WORKTREE_PATH_PREFIX)) goto done; if (!check_meta_file_exists(worktree_path, GOT_WORKTREE_FORMAT)) goto done; if (!remove_worktree(worktree_path)) goto done; ok = 1; done: if (head_ref) got_ref_close(head_ref); if (repo) got_repo_close(repo); return ok; } static int obstruct_meta_file(char **path, const char *worktree_path, const char *name) { FILE *f; char *s = "This file should not be here\n"; int ret = 1; if (asprintf(path, "%s/%s/%s", worktree_path, GOT_WORKTREE_GOT_DIR, name) == -1) return 0; f = fopen(*path, "w+"); if (f == NULL) { free(*path); return 0; } if (fwrite(s, 1, strlen(s), f) != strlen(s)) { free(*path); ret = 0; } fclose(f); return ret; } static int obstruct_meta_file_and_init(int *ok, struct got_repository *repo, const char *worktree_path, char *name) { const struct got_error *err; char *path; int ret = 0; struct got_reference *head_ref = NULL; if (!obstruct_meta_file(&path, worktree_path, GOT_WORKTREE_FILE_INDEX)) return 0; err = got_ref_open(&head_ref, repo, GOT_REF_HEAD); if (err != NULL || head_ref == NULL) return 0; err = got_worktree_init(worktree_path, head_ref, "/", repo); if (err != NULL && err->code == GOT_ERR_ERRNO && errno == EEXIST) { (*ok)++; ret = 1; } unlink(path); free(path); got_ref_close(head_ref); return ret; } static int worktree_init_exists(const char *repo_path) { const struct got_error *err; struct got_repository *repo = NULL; char worktree_path[PATH_MAX]; char *gotpath = NULL; int ok = 0; err = got_repo_open(&repo, repo_path); if (err != NULL || repo == NULL) goto done; strlcpy(worktree_path, "worktree-XXXXXX", sizeof(worktree_path)); if (mkdtemp(worktree_path) == NULL) goto done; if (mkdir(worktree_path, GOT_DEFAULT_DIR_MODE) == -1 && errno != EEXIST) goto done; if (asprintf(&gotpath, "%s/%s", worktree_path, GOT_WORKTREE_GOT_DIR) == -1) goto done; if (mkdir(gotpath, GOT_DEFAULT_DIR_MODE) == -1 && errno != EEXIST) goto done; /* Create files which got_worktree_init() will try to create as well. */ if (!obstruct_meta_file_and_init(&ok, repo, worktree_path, GOT_WORKTREE_HEAD)) goto done; if (!obstruct_meta_file_and_init(&ok, repo, worktree_path, GOT_WORKTREE_LOCK)) goto done; if (!obstruct_meta_file_and_init(&ok, repo, worktree_path, GOT_WORKTREE_FILE_INDEX)) goto done; if (!obstruct_meta_file_and_init(&ok, repo, worktree_path, GOT_WORKTREE_REPOSITORY)) goto done; if (!obstruct_meta_file_and_init(&ok, repo, worktree_path, GOT_WORKTREE_PATH_PREFIX)) goto done; if (!obstruct_meta_file_and_init(&ok, repo, worktree_path, GOT_WORKTREE_FORMAT)) goto done; done: if (repo) got_repo_close(repo); free(gotpath); if (ok == 6) remove_worktree(worktree_path); return (ok == 6); } static void process_cb(void *arg, const char *path) { } static int worktree_checkout(const char *repo_path) { const struct got_error *err; struct got_repository *repo = NULL; struct got_reference *head_ref = NULL; struct got_worktree *worktree = NULL; char *makefile_path = NULL, *cfile_path = NULL; char worktree_path[PATH_MAX]; int ok = 0; struct stat sb; err = got_repo_open(&repo, repo_path); if (err != NULL || repo == NULL) goto done; err = got_ref_open(&head_ref, repo, GOT_REF_HEAD); if (err != NULL || head_ref == NULL) goto done; strlcpy(worktree_path, "worktree-XXXXXX", sizeof(worktree_path)); if (mkdtemp(worktree_path) == NULL) goto done; err = got_worktree_init(worktree_path, head_ref, "/regress/worktree", repo); if (err != NULL) goto done; err = got_worktree_open(&worktree, worktree_path); if (err != NULL) goto done; err = got_worktree_checkout_files(worktree, head_ref, repo, process_cb, NULL); if (err != NULL) goto done; test_printf("checked out %s\n", worktree_path); /* The work tree should contain a Makefile and worktree_test.c. */ if (asprintf(&makefile_path, "%s/Makefile", worktree_path) == -1) goto done; if (stat(makefile_path, &sb) != 0) goto done; else unlink(makefile_path); if (asprintf(&cfile_path, "%s/worktree_test.c", worktree_path) == -1) goto done; if (stat(cfile_path, &sb) != 0) goto done; else unlink(cfile_path); if (!remove_worktree(worktree_path)) goto done; ok = 1; done: if (worktree) got_worktree_close(worktree); if (head_ref) got_ref_close(head_ref); if (repo) got_repo_close(repo); free(makefile_path); free(cfile_path); return ok; } #define RUN_TEST(expr, name) \ { test_ok = (expr); \ printf("test %s %s\n", (name), test_ok ? "ok" : "failed"); \ failure = (failure || !test_ok); } void usage(void) { fprintf(stderr, "usage: worktree_test [-v] [REPO_PATH]\n"); } int main(int argc, char *argv[]) { int test_ok = 0, failure = 0; const char *repo_path; int ch; if (pledge("stdio rpath wpath cpath flock proc", NULL) == -1) err(1, "pledge"); while ((ch = getopt(argc, argv, "v")) != -1) { switch (ch) { case 'v': verbose = 1; break; default: usage(); return 1; } } argc -= optind; argv += optind; if (argc == 0) repo_path = GOT_REPO_PATH; else if (argc == 1) repo_path = argv[0]; else { usage(); return 1; } RUN_TEST(worktree_init(repo_path), "init"); RUN_TEST(worktree_init_exists(repo_path), "init exists"); RUN_TEST(worktree_checkout(repo_path), "checkout"); return failure ? 1 : 0; }