/* -*-c++-*- Copyright (C) 2015 LiveCode Ltd. This file is part of LiveCode. LiveCode is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License v3 as published by the Free Software Foundation. LiveCode 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 General Public License for more details. You should have received a copy of the GNU General Public License along with LiveCode. If not see . */ #include "system-private.h" #include #define MCS_FILE_CONVERT_PATH(path, native_path) \ MCAutoStringRef native_path##__auto; \ MCStringRef native_path; \ do { if (!__MCSFilePathToNative (path, & native_path##__auto)) \ return false; \ native_path = * native_path##__auto; \ } while (0) /* ================================================================ * Error handling * ================================================================ */ bool __MCSFileThrowIOErrorWithErrno (MCStringRef p_native_path, MCStringRef p_message, int p_errno) { MCAutoStringRef t_description; MCAutoNumberRef t_error_code; if (p_errno != 0) { /* UNCHECKED */ MCStringCreateWithCString (strerror (p_errno), &t_description); /* UNCHECKED */ MCNumberCreateWithInteger (p_errno, &t_error_code); } else { t_description = MCSTR("Unknown error"); t_error_code = kMCZero; } MCAutoStringRef t_path; /* UNCHECKED */ __MCSFilePathFromNative (p_native_path, &t_path); if (p_message) { return MCErrorCreateAndThrowWithMessage (kMCSFileIOErrorTypeInfo, p_message, "path", *t_path, "description", *t_description, "error_code", *t_error_code, NULL); } else { return MCErrorCreateAndThrow (kMCSFileIOErrorTypeInfo, "path", *t_path, "description", *t_description, "error_code", *t_error_code, NULL); } } bool __MCSFileThrowReadErrorWithErrno (MCStringRef p_native_path, int p_errno) { return __MCSFileThrowIOErrorWithErrno (p_native_path, MCSTR("Failed to read from file '%{path}': %{description}"), p_errno); } bool __MCSFileThrowWriteErrorWithErrno (MCStringRef p_native_path, int p_errno) { return __MCSFileThrowIOErrorWithErrno (p_native_path, MCSTR("Failed to write to file '%{path}': %{description}"), p_errno); } bool __MCSFileThrowOpenErrorWithErrno (MCStringRef p_native_path, int p_errno) { return __MCSFileThrowIOErrorWithErrno (p_native_path, MCSTR("Failed to open file '%{path}': %{description}"), p_errno); } bool __MCSFileThrowInvalidPathError (MCStringRef p_path) { return MCErrorCreateAndThrow (kMCSFileInvalidPathErrorTypeInfo, "path", p_path, NULL); } /* ================================================================ * Path manipulation * ================================================================ */ MC_DLLEXPORT_DEF bool MCSFileGetCurrentDirectory (MCStringRef & r_path) { MCAutoStringRef t_native_path; if (!__MCSFileGetCurrentDirectory (&t_native_path)) return false; if (!__MCSFilePathFromNative(*t_native_path, r_path)) return false; return true; } MC_DLLEXPORT_DEF bool MCSFilePathIsAbsolute (MCStringRef p_path) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); return __MCSFilePathIsAbsolute (t_native_path); } /* ================================================================ * Whole-file IO * ================================================================ */ MC_DLLEXPORT_DEF bool MCSFileGetContents (MCStringRef p_path, MCDataRef & r_data) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); return __MCSFileGetContents (t_native_path, r_data); } MC_DLLEXPORT_DEF bool MCSFileSetContents (MCStringRef p_path, MCDataRef p_data) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); return __MCSFileSetContents (t_native_path, p_data); } /* ================================================================ * File streams * ================================================================ */ MC_DLLEXPORT_DEF bool MCSFileCreateStream (MCStringRef p_path, intenum_t p_mode, MCStreamRef & r_stream) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); return __MCSFileCreateStream (t_native_path, p_mode, r_stream); } /* ================================================================ * File system operations * ================================================================ */ MC_DLLEXPORT_DEF bool MCSFileDelete (MCStringRef p_path) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); return __MCSFileDelete (t_native_path); } MC_DLLEXPORT_DEF bool MCSFileCreateDirectory (MCStringRef p_path) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); return __MCSFileCreateDirectory (t_native_path); } MC_DLLEXPORT_DEF bool MCSFileDeleteDirectory (MCStringRef p_path) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); return __MCSFileDeleteDirectory (t_native_path); } /* This callback function is used by MCSFileGetDirectoryEntries() to * convert directory entries from system filename representation to * LiveCode's internal representation. */ static bool MCSFileGetDirectoryEntries_MapCallback (void *p_context, MCValueRef p_native_path, MCValueRef & r_path) { MCAssert (kMCStringTypeInfo == MCValueGetTypeInfo (p_native_path)); MCStringRef t_path; if (!__MCSFilePathFromNative (static_cast(p_native_path), t_path)) return false; r_path = t_path; return true; } MC_DLLEXPORT_DEF bool MCSFileGetDirectoryEntries (MCStringRef p_path, MCProperListRef & r_entries) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); MCAutoProperListRef t_native_entries; if (!__MCSFileGetDirectoryEntries (t_native_path, &t_native_entries)) return false; /* Convert the returned directory entries to LiveCode path * representation. */ return MCProperListMap (*t_native_entries, MCSFileGetDirectoryEntries_MapCallback, r_entries, NULL); } MC_DLLEXPORT_DEF bool MCSFileGetType (MCStringRef p_path, bool p_follow_links, MCSFileType & r_type) { MCS_FILE_CONVERT_PATH(p_path, t_native_path); bool t_success = true; if (t_success) { t_success = __MCSFileGetType (t_native_path, p_follow_links, r_type); } if (t_success && r_type == kMCSFileTypeUnsupported) { MCLog ("%s: file '%@' has unrecognised type", __FUNCTION__, p_path); } return t_success; } /* ================================================================ * Initialization * ================================================================ */ MC_DLLEXPORT_DEF MCTypeInfoRef kMCSFileIOErrorTypeInfo; MC_DLLEXPORT_DEF MCTypeInfoRef kMCSFileEndOfFileErrorTypeInfo; MC_DLLEXPORT_DEF MCTypeInfoRef kMCSFileInvalidPathErrorTypeInfo; bool __MCSFileInitialize (void) { /* Create error types */ if (!MCNamedErrorTypeInfoCreate( MCNAME("livecode.lang.FileIOError"), MCNAME("file"), MCSTR("File input/output error for '%{path}': %{description}"), kMCSFileIOErrorTypeInfo)) return false; if (!MCNamedErrorTypeInfoCreate ( MCNAME("livecode.lang.EndOfFileError"), MCNAME("file"), MCSTR("End of file '%{path}'"), kMCSFileEndOfFileErrorTypeInfo)) return false; if (!MCNamedErrorTypeInfoCreate ( MCNAME("livecode.lang.InvalidFilenameError"), MCNAME("file"), MCSTR("No valid native path representation for path '%{path}'"), kMCSFileInvalidPathErrorTypeInfo)) return false; return true; } void __MCSFileFinalize (void) { MCValueRelease (kMCSFileInvalidPathErrorTypeInfo); MCValueRelease (kMCSFileEndOfFileErrorTypeInfo); MCValueRelease (kMCSFileIOErrorTypeInfo); }