:orphan: .. index:: pair: group; File operations .. _doxid-group__te__tools__te__file: File operations =============== .. toctree:: :hidden: Overview ~~~~~~~~ Functions to operate the files. :ref:`More...` .. ref-code-block:: cpp :class: doxyrest-overview-code-block // typedefs typedef :ref:`te_errno` :ref:`te_file_scandir_callback`( const char *pattern, const char *pathname, void *data ); // global functions char* :ref:`te_basename`(const char* pathname); char* :ref:`te_readlink_fmt`(const char* path_fmt, ...); char char* :ref:`te_dirname`(const char* pathname); char* :ref:`te_file_join_filename`(:ref:`te_string`* dest, const char* dirname, const char* path, const char* suffix); int :ref:`te_file_create_unique_fd_va`(char** filename, const char* prefix_format, const char* suffix, va_list ap); int :ref:`te_file_create_unique_fd`(char** filename, const char* prefix_format, const char* suffix, ...); int char* :ref:`te_file_create_unique`(const char* prefix_format, const char* suffix, ...); int char FILE* :ref:`te_fopen_fmt`(const char* mode, const char* path_fmt, ...); int char FILE pid_t :ref:`te_file_read_pid`(const char* pid_path); :ref:`te_errno` :ref:`te_file_resolve_pathname_vec`(const char* filename, const :ref:`te_vec`* pathvec, int mode, char** resolved); :ref:`te_errno` :ref:`te_file_resolve_pathname`(const char* filename, const char* path, int mode, const char* basename, char** resolved); :ref:`te_errno` :ref:`te_file_check_executable`(const char* path); :ref:`te_errno` :ref:`te_access_fmt`(int mode, const char* fmt, ...); :ref:`te_errno` :ref:`te_errno` :ref:`te_unlink_fmt`(const char* fmt, ...); :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_file_read_string`(:ref:`te_string`* dest, bool binary, size_t maxsize, const char* path_fmt, ...); :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_file_write_string`(const :ref:`te_string`* src, size_t fitlen, int flags, mode_t mode, const char* path_fmt, ...); :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_file_read_text`(const char* path, char* buffer, size_t bufsize); :ref:`te_errno` :ref:`te_file_scandir`(const char* dirname, :ref:`te_file_scandir_callback`* callback, void* data, const char* pattern_fmt, ...); :ref:`te_errno` char* :ref:`te_file_extract_glob`(const char* filename, const char* pattern, bool basename); .. _details-group__te__tools__te__file: Detailed Documentation ~~~~~~~~~~~~~~~~~~~~~~ Functions to operate the files. Copyright (C) 2018-2023 OKTET Labs Ltd. All rights reserved. Typedefs -------- .. index:: pair: typedef; te_file_scandir_callback .. _doxid-group__te__tools__te__file_1ga20b6fe0a02c9cc4a2b6593d43e25c506: .. ref-code-block:: cpp :class: doxyrest-title-code-block typedef :ref:`te_errno` te_file_scandir_callback( const char *pattern, const char *pathname, void *data ) Function type for callbacks for :ref:`te_file_scandir() `. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - pattern - pattern used to filter pathnames * - pathname - full pathname of a current file * - data - user data as passed to :ref:`te_file_scandir() `. * - TE_EOK - The scanning would stop, but ``0`` would be returned by :ref:`te_file_scandir() `. .. rubric:: Returns: status code Global Functions ---------------- .. index:: pair: function; te_basename .. _doxid-group__te__tools__te__file_1gabfddc0e293d05cfc88893d1260bd9384: .. ref-code-block:: cpp :class: doxyrest-title-code-block char* te_basename(const char* pathname) Get a basename from ``pathname``, and check if it is valid. Unlike system basename() it does not modify the contents of ``pathname`` .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - pathname - File pathname .. rubric:: Returns: A heap-allocated string containing an extracted file name, or ``NULL`` if ``pathname`` is invalid, or allocation fails .. index:: pair: function; te_readlink_fmt .. _doxid-group__te__tools__te__file_1gaff0f37c1fe0aaf27e9a0845091f2bd50: .. ref-code-block:: cpp :class: doxyrest-title-code-block char* te_readlink_fmt(const char* path_fmt, ...) Get a resolved link using ``path``. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - path_fmt - Path format string. * - ... - Format string arguments. .. rubric:: Returns: A heap-allocated string containing a resolved link, or ``NULL`` if an error has occurred. .. index:: pair: function; te_dirname .. _doxid-group__te__tools__te__file_1gad286e1d4550d546b382048763008e78b: .. ref-code-block:: cpp :class: doxyrest-title-code-block char char* te_dirname(const char* pathname) Get a dirname from ``pathname``, and check if it is valid. Unlike system dirname() it does not modify the contents of ``pathname`` .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - pathname - File pathname .. rubric:: Returns: A heap-allocated string containing an extracted directory name, or ``NULL`` if ``pathname`` is invalid, or allocation fails .. index:: pair: function; te_file_join_filename .. _doxid-group__te__tools__te__file_1ga4e141d3a1e6da0296759baf45647606c: .. ref-code-block:: cpp :class: doxyrest-title-code-block char* te_file_join_filename(:ref:`te_string`* dest, const char* dirname, const char* path, const char* suffix) Construct a filename from components. Unlike :ref:`te_file_resolve_pathname() `, the function does a purely syntactic operation, so no checks are made that a pathname exists or is accessible. Therefore it is suitable for constructing filenames on one host that will be used on another. The constructing algorithm is as follows: * if ``path`` is ``NULL``, it is treated as an empty string; * if ``dirname`` is ``NULL`` or ``path`` is an absolute path, ``path`` is used as is; * otherwise, ``dirname`` and ``path`` are joined using ``/`` separator; * then if ``suffix`` is not ``NULL``, it is appended to the filename obtained at the previous step; if that name ends with a ``/``, it is first stripped; i.e. a suffix never creates a new pathname component. If ``dest`` is ``NULL``, a fresh string is allocated and returned. The name is appended to the string contents. If ``dest`` is not empty upon entry and ``dirname`` is empty, an extra slash may be added before the resulting pathname. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - dest - the string to hold the name or ``NULL``. * - dirname - directory name * - path - pathname (relative or absolute) * - suffix - suffix .. rubric:: Returns: the pointer to the contents of ``dest`` or a heap-allocated buffer .. index:: pair: function; te_file_create_unique_fd_va .. _doxid-group__te__tools__te__file_1gaf5f9d9c78b82106375299f9a90617e52: .. ref-code-block:: cpp :class: doxyrest-title-code-block int te_file_create_unique_fd_va(char** filename, const char* prefix_format, const char* suffix, va_list ap) Create a file of unique name. It builds a template of file name in mkstemps compatible format which has the form "prefixXXXXXXsuffix", or "prefixXXXXXX" if ``suffix`` is ``NULL`` .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - filename - Name of created file. On success it should be freed with free(3) when it is no longer needed * - prefix_format - Format string of filename prefix * - suffix - Filename suffix, can be ``NULL`` * - ap - List of arguments .. rubric:: Returns: File descriptor of the created file, or ``-1`` in case of error .. rubric:: See also: :ref:`te_file_create_unique_fd `, :ref:`te_file_create_unique ` .. index:: pair: function; te_file_create_unique_fd .. _doxid-group__te__tools__te__file_1ga5d894dec15a053a33a1282164535cff7: .. ref-code-block:: cpp :class: doxyrest-title-code-block int te_file_create_unique_fd(char** filename, const char* prefix_format, const char* suffix, ...) Create a file of unique name, see details in description of :ref:`te_file_create_unique_fd_va() ` .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - filename - Name of created file. On success it should be freed with free(3) when it is no longer needed * - prefix_format - Format string of filename prefix * - suffix - Filename suffix, can be ``NULL`` * - ... - Format string arguments .. rubric:: Returns: File descriptor of the created file, or ``-1`` in case of error .. rubric:: See also: :ref:`te_file_create_unique_fd_va `, :ref:`te_file_create_unique ` .. index:: pair: function; te_file_create_unique .. _doxid-group__te__tools__te__file_1gafff4d7b107a87b63c8708620c0354773: .. ref-code-block:: cpp :class: doxyrest-title-code-block int char* te_file_create_unique(const char* prefix_format, const char* suffix, ...) Create a file of unique name, see details in description of :ref:`te_file_create_unique_fd_va() ` Return value should be freed with free(3) when it is no longer needed .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - prefix_format - Format string of filename prefix * - suffix - Filename suffix, can be ``NULL`` * - ... - Format string arguments .. rubric:: Returns: Name of created file, or ``NULL`` in case of error .. rubric:: See also: :ref:`te_file_create_unique_fd_va `, :ref:`te_file_create_unique_fd ` .. index:: pair: function; te_fopen_fmt .. _doxid-group__te__tools__te__file_1ga5f6a832c34aab36f7c0dbedc6a756a80: .. ref-code-block:: cpp :class: doxyrest-title-code-block int char FILE* te_fopen_fmt(const char* mode, const char* path_fmt, ...) Call fopen() with a path specified by a format string and arguments. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - mode - Opening mode string. * - path_fmt - File path format string. * - ... - Format string arguments. .. rubric:: Returns: FILE pointer on success, ``NULL`` on failure. .. index:: pair: function; te_file_read_pid .. _doxid-group__te__tools__te__file_1ga8b0873eea87957d5e7ed2b1868479a6f: .. ref-code-block:: cpp :class: doxyrest-title-code-block int char FILE pid_t te_file_read_pid(const char* pid_path) Read process identifier from PID file .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - pid_path - Path to PID file .. rubric:: Returns: Process ID or ``-1`` in case of error .. index:: pair: function; te_file_resolve_pathname_vec .. _doxid-group__te__tools__te__file_1gacc7bf38c3519ac08191e430af9e3d141: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` te_file_resolve_pathname_vec(const char* filename, const :ref:`te_vec`* pathvec, int mode, char** resolved) Search a relative filename in a vector of directories. If the filename is absolute (i.e. starts with ``/``), it is checked and no search is performed. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - filename - Filename to check * - pathvec - Vector of directories * - mode - Access mode to check (as used by system access()) * - resolved - A location for a resolved pathname (which must be free()'d. If ``NULL``, the function just checks the presence of ``filename`` somewhere in the ``path``. .. rubric:: Returns: Status code (possible values match those reported by access()) .. index:: pair: function; te_file_resolve_pathname .. _doxid-group__te__tools__te__file_1ga1947fd6fa5bda77c92ddbd8fb0c1623e: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` te_file_resolve_pathname(const char* filename, const char* path, int mode, const char* basename, char** resolved) Search a relative filename in a colon-separated list of directories. The function is like :ref:`te_file_resolve_pathname_vec() `, only it takes a colon-separated string for a path instead of a prepared vector. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - filename - Filename to check * - path - Colon-separated list of directories * - mode - Access mode to check (as used by system access()) * - basename - If not ``NULL``, ``filename`` is first looked up in a directory part of ``basename`` (i.e. if ``basename`` points to a directory, it is used, otherwise the result of dirname() is used). * - resolved - A location for a resolved pathname (which must be free()'d. If ``NULL``, the function just checks the presence of ``filename`` somewhere in the ``path``. .. rubric:: Returns: Status code (possible values match those reported by access()) .. index:: pair: function; te_file_check_executable .. _doxid-group__te__tools__te__file_1ga4b2436913fdcd1895257261d8679c4fb: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` te_file_check_executable(const char* path) Check that the file is executable If ``path`` does't contains '/' then ``path`` is concatenated with the environment variable PATH. Otherwise, we are looking for a file from the current location .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - path - Path to the file .. rubric:: Returns: Status code .. index:: pair: function; te_access_fmt .. _doxid-group__te__tools__te__file_1ga4ce1003d604ceeadbf7bab43330f4cfe: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` te_access_fmt(int mode, const char* fmt, ...) Check that a filename is accessible for a given ``mode``. The filename is constructed from the format string and arguments. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - mode - access mode (see system access()) * - fmt - format string * - ... - arguments * - TE_EACCESS - Access to the file is denied * - TE_ENOENT - The file or any of its parents does not exist .. rubric:: Returns: status code .. index:: pair: function; te_unlink_fmt .. _doxid-group__te__tools__te__file_1ga10ab43abc5bc13254003a2de2603bac9: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` :ref:`te_errno` te_unlink_fmt(const char* fmt, ...) Delete a file. The filename is constructed from the format string and arguments. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - fmt - format string * - ... - arguments * - TE_ENOENT - The file or any of its parents does not exist. .. rubric:: Returns: status code .. index:: pair: function; te_file_read_string .. _doxid-group__te__tools__te__file_1ga40e514272c3536f63887304dfe6489f0: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` te_file_read_string(:ref:`te_string`* dest, bool binary, size_t maxsize, const char* path_fmt, ...) Read the contents of the file into a dynamic string ``dest``. If ``binary`` is ``false``, the function ensures that there is no embedded zeroes in the file content and strips off trailing newlines if there are any. If ``maxsize`` is not zero, the function ensures that the file size is not greater than it. That allows to avoid crashes if ``dest`` is a statically allocated TE string and in general to prevent memory bombs if a file comes from an untrusted source. The function should be used with caution on non-regular files such as named FIFOs or character devices, as it may block indefinitely or return partial data. Using it with regular files on special filesystems such as ``/proc`` is completely reliable. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - dest - TE string (the content of the file will be appended to it) * - binary - reading mode (binary or text mode) * - maxsize - the maximum allowed file size, if not zero * - path_fmt - pathname format string * - ... - arguments * - TE_EFBIG - The file size is larger than ``maxsize``. * - TE_EILSEQ - The file contains embedded zeroes and ``binary`` is ``false``. .. rubric:: Returns: status code .. index:: pair: function; te_file_write_string .. _doxid-group__te__tools__te__file_1ga4ac6b6c3cfb36612372dd5f5efd2c201: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` te_file_write_string(const :ref:`te_string`* src, size_t fitlen, int flags, mode_t mode, const char* path_fmt, ...) Write the contents of the file from a dynamic string ``src``. The file is opened with POSIX ``flags`` ORed with ``O_WRONLY`` and if the file is created, then access ``mode`` will be set on it. If ``fitlen`` is not zero, the resulting file will be exactly that long: * if ``src`` is longer than ``fitlen``, it will be truncated; * if ``src`` is shorter, it will be repeated until the length is reached. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - src - source TE string * - fitlen - desired file length (ignored if zero) * - flags - POSIX open flags, such as ``O_CREAT`` * - mode - POSIX access mode for newly created files * - path_fmt - pathname format string * - ... - arguments .. rubric:: Returns: status code .. index:: pair: function; te_file_read_text .. _doxid-group__te__tools__te__file_1gae8ba87b52ef51157ab22a7bea5ba6365: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` :ref:`te_errno` te_file_read_text(const char* path, char* buffer, size_t bufsize) Read the contents of the file ``path`` into ``buffer``. The function ensures that the contents of the file is no larger than ``bufsize`` minus one and that the contents contain no embedded zeroes. The resulting string will be zero-terminated. If there are trailing newlines, they are stripped off. The file should be random-access, so the function cannot read from sockets, named FIFOs and most kinds of character devices. Deprecated Prefer :ref:`te_file_read_string() `. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - path - Pathname to read * - buffer - Destination buffer * - bufsize - The size of ``buffer`` including terminating zero. * - TE_EFBIG - The file size is larger than ``bufsize`` * - TE_EILSEQ - The file contains embedded zeroes .. rubric:: Returns: Status code .. index:: pair: function; te_file_scandir .. _doxid-group__te__tools__te__file_1gaa9de31e796cf8ffb6e97c0c12ca27010: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` te_file_scandir(const char* dirname, :ref:`te_file_scandir_callback`* callback, void* data, const char* pattern_fmt, ...) Call ``callback`` for each file in ``dirname`` matching a pattern. Special entries ````. and ````.. are always skipped. The callback is passed a concatenation of a ``dirname`` and the basename of the current file, not just the basename as scandir() would do. If the callback returns non-zero, the scanning stop and the result value is passed upstream, but if it is ``TE_EOK``, zero would be returned. ``pattern_fmt`` may be ``NULL``, in which case no arguments shall be be provided and all files in ``dirname`` will be processed. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - dirname - directory to scan * - callback - the callback * - data - user data passed to ``callback`` * - pattern_fmt - glob-style pattern format * - ... - arguments .. rubric:: Returns: status code .. index:: pair: function; te_file_extract_glob .. _doxid-group__te__tools__te__file_1gaff45ec22f4cd80739eeee98c8681c9fc: .. ref-code-block:: cpp :class: doxyrest-title-code-block :ref:`te_errno` char* te_file_extract_glob(const char* filename, const char* pattern, bool basename) Extract a varying part of a ``filename`` matching ``pattern``. Only a limited subset of globs is supported, namely it must contain exactly one ``*`` wildcard, otherwise a fatal error is reported. .. rubric:: Parameters: .. list-table:: :widths: 20 80 * - filename - filename * - pattern - glob pattern with a single ``*`` * - basename - if ``true``, only the basename of ``filename`` is used for matching. .. rubric:: Returns: the varying part of the filename (must be free()'d) or ``NULL`` if ``filename`` does not match a ``pattern``.