dav/system.c

Fri, 02 Aug 2019 22:04:00 +0200

author
Olaf Wintermann <olaf.wintermann@gmail.com>
date
Fri, 02 Aug 2019 22:04:00 +0200
changeset 611
a7c48e0dca88
parent 608
3e4c0285a868
child 612
66dc8b992d8d
permissions
-rw-r--r--

implement links on Windows (shelllink)

/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright 2019 Olaf Wintermann. All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 *   1. Redistributions of source code must retain the above copyright
 *      notice, this list of conditions and the following disclaimer.
 *
 *   2. Redistributions in binary form must reproduce the above copyright
 *      notice, this list of conditions and the following disclaimer in the
 *      documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER 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 <libidav/utils.h>

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <dirent.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <errno.h>

#include <ucx/string.h>

#ifndef _WIN32
#include <unistd.h>
#endif

#include "system.h"

void sys_freedirent(SysDirEnt *ent) {
    free(ent->name);
    free(ent);
}

#ifndef _WIN32
/* ---------- POSIX implementation ---------- */

void sys_init(void) {
    
}
void sys_uninit(void) {
    
}

SYS_DIR sys_opendir(const char *path) {
    DIR *dir = opendir(path);
    if(!dir) {
        return NULL;
    }
    SysDir *d = malloc(sizeof(SysDir));
    d->dir = dir;
    d->ent = NULL;
    return d;
}

SysDirEnt* sys_readdir(SYS_DIR dir) {
    if(dir->ent) {
        free(dir->ent->name);
        free(dir->ent);
        dir->ent = NULL;
    }
    struct dirent *ent = readdir(dir->dir);
    if(ent) {
        SysDirEnt *e = malloc(sizeof(SysDirEnt));
        e->name = strdup(ent->d_name);
        dir->ent = e;
        return e;
    }
    return NULL;
}

void sys_closedir(SYS_DIR dir) {
    closedir(dir->dir);
    if(dir->ent) {
        free(dir->ent->name);
        free(dir->ent);
    }
    free(dir);
}

FILE* sys_fopen(const char *path, const char *mode) {
    return fopen(path, mode);
}

int sys_stat(const char *path, SYS_STAT *s) {
    return stat(path, s);
}

int sys_lstat(const char *path, SYS_STAT *s) {
    return lstat(path, s);
}

int sys_islink(const char *path) {
    struct stat s;
    if(!lstat(path, &s)) {
        return S_ISLNK(s.st_mode);
    }
}   return 0;

int sys_rename(const char *oldpath, const char *newpath) {
    return rename(oldpath, newpath);
}

int sys_unlink(const char *path) {
    return unlink(path);
}

int sys_mkdir(const char *path) {
    return mkdir(path, S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH);
}

char* sys_readlink(const char *path, SYS_STAT *s) {
    char *ret = NULL;
    
    off_t l_sz = s->st_size + 16;
    size_t lnksize = l_sz > 256 ? l_sz : 256;
    char *lnkbuf = malloc(lnksize);
    
    ssize_t len = 0;
    for(int i=0;i<4;i++) {
        // we try to read the link at most 4 times
        // only repeat if the buffer is too small
        len = readlink(path, lnkbuf, lnksize);
        if(len < lnksize) {
            ret = lnkbuf; // success
            lnkbuf[len] = 0; // terminate buffer
            break;
        }
        lnksize *= 2; // retry with bigger buffer
        lnkbuf = realloc(lnkbuf, lnksize);
    }
    
    if(!ret) {
        free(lnkbuf);
    }
    return ret;
}

int sys_symlink(const char *target, const char *linkpath) {
    int err = symlink(target, linkpath);
    if(err && errno == EEXIST) {
        if(unlink(linkpath)) {
            return 1;
        }
        return sys_symlink(target, linkpath);
    }
    return err;
}

#else
/* ---------- Windows implementation ---------- */

#include <windows.h>
#include <winnls.h>
#include <shobjidl.h>
#include <objbase.h>
#include <objidl.h>

#include <direct.h>
#include <wchar.h>

void sys_init(void) {
    HRESULT res =  CoInitialize(NULL);
    if(res != S_OK) {
        fprintf(stderr, "Error: CoInitialize failed\n");
    }
}

void sys_uninit(void) {
    CoUninitialize();
}

static wchar_t* path2winpath(const char *path, int dir, int *newlen) {
    size_t len = strlen(path);
    size_t lenadd = dir ? 2 : 0;
    
    
    wchar_t *wpath = calloc(len+lenadd+1, sizeof(wchar_t));
    int wlen = MultiByteToWideChar(
            CP_UTF8,
            0,
            path,
            len,
            wpath,
            len+1
            );
    if(newlen) {
        *newlen = wlen;
    }
    for(int i=0;i<wlen;i++) {
        if(wpath[i] == L'/') {
            wpath[i] = L'\\';
        }
    }
    
    if(dir) {
        if(wpath[wlen-1] != L'\\') {
            wpath[wlen++] = L'\\';
        }
        wpath[wlen++] = L'*';
    }
    wpath[wlen] = 0;
    
    return wpath;
}

static char* winpath2multibyte(const wchar_t *wpath, size_t wlen) {
    size_t maxlen = wlen * 4;
    char *ret = malloc(maxlen + 1);
    int ret_len = WideCharToMultiByte(
        CP_UTF8,
        0,
        wpath,
        wlen,
        ret,
        maxlen,
        NULL,
        NULL);
    ret[ret_len] = 0;
    return ret;
}



SYS_DIR sys_opendir(const char *path) {
    struct WinDir *dir = malloc(sizeof(struct WinDir));
    wchar_t *dirpath = path2winpath(path, TRUE, NULL);
    if(!dirpath) {
        fprintf(stderr, "Cannot convert path \"%s\" to UTF16\n", path);
        free(dir);
        return NULL;
    }
    dir->first = 1;
    dir->handle = FindFirstFileW(dirpath, &dir->finddata);
    free(dirpath);
    if(dir->handle == INVALID_HANDLE_VALUE) {
        free(dir);
        return NULL;
    }
    dir->ent = NULL;
    return dir;
}

SysDirEnt* sys_readdir(SYS_DIR dir) {
    if(dir->ent) {
        free(dir->ent->name);
        free(dir->ent);
        dir->ent = NULL;
    }
    if(dir->first) {
        dir->first = 0;
    } else {
        if(FindNextFileW(dir->handle, &dir->finddata) == 0) {
            return NULL;
        }
    }
    
    size_t namelen = wcslen(dir->finddata.cFileName);
    
    char *name = malloc((namelen+1)*4);
    int nlen = WideCharToMultiByte(
            CP_UTF8,
            0,
            dir->finddata.cFileName,
            -1,
            name,
            256,
            NULL,
            NULL);
    if(nlen > 0) {
        name[nlen] = 0;
        SysDirEnt *ent = malloc(sizeof(SysDirEnt));
        ent->name = name;
        dir->ent = ent;
        return ent;
    } else {
        return NULL;
    }
}

void sys_closedir(SYS_DIR dir) {
    if(dir->ent) {
        free(dir->ent->name);
        free(dir->ent);
    }
    FindClose(dir->handle);
    free(dir);
}

FILE* sys_fopen(const char *path, const char *mode) {
    wchar_t *fpath = path2winpath(path, FALSE, NULL); 
    wchar_t *fmode = path2winpath(mode, FALSE, NULL);
    
    FILE *file = (fpath && fmode) ? _wfopen(fpath, fmode) : NULL;
    free(fpath);
    free(fmode);
    return file;
}

int sys_stat(const char *path, SYS_STAT *s) {
    wchar_t *fpath = path2winpath(path, FALSE, NULL);
    if(!fpath) {
        fprintf(stderr, "Cannot convert path \"%s\" to UTF16\n", path);
        return -1;
    }
    int ret = _wstat64(fpath, s);
    free(fpath);
    return ret;
}

int sys_lstat(const char *path, SYS_STAT *s) {
    return sys_stat(path, s); // unsupported on windows
}

int sys_islink(const char *path) {
    // don't use symlinks on windows, because it is not really useful
    // however, we interpret .lnk files as symlinks
    int ret = 0;
    
    scstr_t path_s = scstr(path);
    if(scstrsuffix(path_s, SC(".lnk"))) {
        // looks like a .lnk file
        // check content
        IShellLink *sl;
        HRESULT hres;
        hres = CoCreateInstance(&CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, &IID_IShellLinkW, (LPVOID*)&sl);
        if(SUCCEEDED(hres)) { 
            IPersistFile *file;
            hres = sl->lpVtbl->QueryInterface(sl, &IID_IPersistFile, (void**)&file);
            if(!SUCCEEDED(hres)) {
                sl->lpVtbl->Release(sl);
                return ret;
            }
            
            int newlen = 0;
            wchar_t *wpath = path2winpath(path, 0, &newlen);
            
            hres = file->lpVtbl->Load(file, wpath, STGM_READ);
            if(SUCCEEDED(hres)) {
                ret = 1;
                file->lpVtbl->Release(file);
            }
            free(wpath);
            
            sl->lpVtbl->Release(sl);
        }
    }
    return ret;
}

int sys_rename(const char *oldpath, const char *newpath) {
    wchar_t *o = path2winpath(oldpath, FALSE, NULL);
    wchar_t *n = path2winpath(newpath, FALSE, NULL);
    if(!o || !n) {
        return -1;
    }
    
    struct __stat64 s;
    if(!_wstat64(n, &s)) {
        if(_wunlink(n)) {
            fprintf(stderr, "sys_rename: cannot delete existing file: %ls\n", n);
        }
    }
    
    int ret = _wrename(o, n);
    free(o);
    free(n);
    return ret;
}

int sys_unlink(const char *path) {
    wchar_t *wpath = path2winpath(path, FALSE, NULL);
    if(!wpath) {
        fprintf(stderr, "sys_unlink: cannot convert path\n");
        return -1;
    }
    int ret = _wunlink(wpath);
    free(wpath);
    return ret;
}

int sys_mkdir(const char *path) {
    wchar_t *wpath = path2winpath(path, FALSE, NULL);
    if(!wpath) {
        fprintf(stderr, "sys_mkdir: cannot convert path\n");
        return -1;
    }
    int ret = _wmkdir(wpath);
    free(wpath);
    return ret;
}

char* sys_readlink(const char *path, SYS_STAT *s) {
    char *ret_link = NULL;
    
    // create COM object for using the ShellLink interface
    IShellLinkW *sl;
    HRESULT hres = CoCreateInstance(&CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, &IID_IShellLinkW, (LPVOID*)&sl);
    if(!SUCCEEDED(hres)) {
        return NULL;
    }
    
    IPersistFile *file;
    hres = sl->lpVtbl->QueryInterface(sl, &IID_IPersistFile, (void**)&file);
    if(!SUCCEEDED(hres)) {
        sl->lpVtbl->Release(sl);
        return NULL;
    }
    
    // load .lnk file
    int newlen = 0;
    wchar_t *wpath = path2winpath(path, 0, &newlen);
    hres = file->lpVtbl->Load(file, wpath, STGM_READ);
    if(SUCCEEDED(hres)) {
        WCHAR link_path[MAX_PATH];
        memset(link_path, 0, MAX_PATH);
        
        hres = sl->lpVtbl->Resolve(sl, 0, SLR_NO_UI);
        if(SUCCEEDED(hres)) {
            hres = sl->lpVtbl->GetPath(sl, link_path, MAX_PATH, NULL, SLGP_SHORTPATH);
            if(SUCCEEDED(hres)) {
                ret_link = winpath2multibyte(link_path, wcslen(link_path));
            }
        }
    }
    // cleanup
    free(wpath);
    file->lpVtbl->Release(file);
    sl->lpVtbl->Release(sl);
    
    return ret_link;
}

int sys_symlink(const char *target, const char *linkpath) {
    // convert relative target to absolut path
    char *link_parent = util_parent_path(linkpath);
    char *target_unnormalized = util_concat_path(link_parent, target);
    char *target_normalized = util_path_normalize(target_unnormalized);
    
    free(link_parent);
    free(target_unnormalized);
    
    // convert to wchar_t*
    int wtargetlen = 0;
    wchar_t *wtarget = path2winpath(target_normalized, FALSE, &wtargetlen);
    free(target_normalized);
    if(!wtarget) {
        return 1;
    }
    
    int wlinkpathlen = 0;
    wchar_t *wlinkpath = path2winpath(linkpath, FALSE, &wlinkpathlen);
    if(!wlinkpath) {
        free(wtarget);
        return 1;
    }
    
    int ret = 1;
    
    // create COM object for using the ShellLink interface
    IShellLinkW *sl;
    HRESULT hres = CoCreateInstance(&CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, &IID_IShellLinkW, (LPVOID*)&sl);
    if(SUCCEEDED(hres)) {    
        IPersistFile *file;
        hres = sl->lpVtbl->QueryInterface(sl, &IID_IPersistFile, (void**)&file);
        if(SUCCEEDED(hres)) {
            // try to load the shortcut
            file->lpVtbl->Load(file, wlinkpath, STGM_READ); // ignore error
            
            // set path
            hres = sl->lpVtbl->SetPath(sl, wtarget);
            if(SUCCEEDED(hres)) {
                hres = file->lpVtbl->Save(file, wlinkpath, TRUE);
                if(SUCCEEDED(hres)) {
                    // successfully created/modified shortcut
                    ret = 0; // ok
                }
            }
            
            file->lpVtbl->Release(file);
        }
        
        sl->lpVtbl->Release(sl);
    }

    free(wtarget);
    free(wlinkpath);
    
    return ret;
}

#endif

mercurial