libqaeda

Unnamed repository; edit this file 'description' to name the repository.
Info | Log | Files | Refs | README | LICENSE

cwalk.h (19488B)


      1 #pragma once
      2 
      3 #ifndef CWK_LIBRARY_H
      4 #define CWK_LIBRARY_H
      5 
      6 #include <stdbool.h>
      7 #include <stddef.h>
      8 
      9 #if defined(_WIN32) || defined(__CYGWIN__)
     10 #define CWK_EXPORT __declspec(dllexport)
     11 #define CWK_IMPORT __declspec(dllimport)
     12 #elif __GNUC__ >= 4
     13 #define CWK_EXPORT __attribute__((visibility("default")))
     14 #define CWK_IMPORT __attribute__((visibility("default")))
     15 #else
     16 #define CWK_EXPORT
     17 #define CWK_IMPORT
     18 #endif
     19 
     20 #if defined(CWK_SHARED)
     21 #if defined(CWK_EXPORTS)
     22 #define CWK_PUBLIC CWK_EXPORT
     23 #else
     24 #define CWK_PUBLIC CWK_IMPORT
     25 #endif
     26 #else
     27 #define CWK_PUBLIC
     28 #endif
     29 
     30 #ifdef __cplusplus
     31 extern "C"
     32 {
     33 #endif
     34 
     35 /**
     36  * A segment represents a single component of a path. For instance, on linux a
     37  * path might look like this "/var/log/", which consists of two segments "var"
     38  * and "log".
     39  */
     40 struct cwk_segment
     41 {
     42   const char *path;
     43   const char *segments;
     44   const char *begin;
     45   const char *end;
     46   size_t size;
     47 };
     48 
     49 /**
     50  * The segment type can be used to identify whether a segment is a special
     51  * segment or not.
     52  *
     53  * CWK_NORMAL - normal folder or file segment
     54  * CWK_CURRENT - "./" current folder segment
     55  * CWK_BACK - "../" relative back navigation segment
     56  */
     57 enum cwk_segment_type
     58 {
     59   CWK_NORMAL,
     60   CWK_CURRENT,
     61   CWK_BACK
     62 };
     63 
     64 /**
     65  * @brief Determines the style which is used for the path parsing and
     66  * generation.
     67  */
     68 enum cwk_path_style
     69 {
     70   CWK_STYLE_WINDOWS,
     71   CWK_STYLE_UNIX
     72 };
     73 
     74 /**
     75  * @brief Generates an absolute path based on a base.
     76  *
     77  * This function generates an absolute path based on a base path and another
     78  * path. It is guaranteed to return an absolute path. If the second submitted
     79  * path is absolute, it will override the base path. The result will be
     80  * written to a buffer, which might be truncated if the buffer is not large
     81  * enough to hold the full path. However, the truncated result will always be
     82  * null-terminated. The returned value is the amount of characters which the
     83  * resulting path would take if it was not truncated (excluding the
     84  * null-terminating character).
     85  *
     86  * @param base The absolute base path on which the relative path will be
     87  * applied.
     88  * @param path The relative path which will be applied on the base path.
     89  * @param buffer The buffer where the result will be written to.
     90  * @param buffer_size The size of the result buffer.
     91  * @return Returns the total amount of characters of the new absolute path.
     92  */
     93 CWK_PUBLIC size_t cwk_path_get_absolute(const char *base, const char *path,
     94   char *buffer, size_t buffer_size);
     95 
     96 /**
     97  * @brief Generates a relative path based on a base.
     98  *
     99  * This function generates a relative path based on a base path and another
    100  * path. It determines how to get to the submitted path, starting from the
    101  * base directory. The result will be written to a buffer, which might be
    102  * truncated if the buffer is not large enough to hold the full path. However,
    103  * the truncated result will always be null-terminated. The returned value is
    104  * the amount of characters which the resulting path would take if it was not
    105  * truncated (excluding the null-terminating character).
    106  *
    107  * @param base_directory The base path from which the relative path will
    108  * start.
    109  * @param path The target path where the relative path will point to.
    110  * @param buffer The buffer where the result will be written to.
    111  * @param buffer_size The size of the result buffer.
    112  * @return Returns the total amount of characters of the full path.
    113  */
    114 CWK_PUBLIC size_t cwk_path_get_relative(const char *base_directory,
    115   const char *path, char *buffer, size_t buffer_size);
    116 
    117 /**
    118  * @brief Joins two paths together.
    119  *
    120  * This function generates a new path by combining the two submitted paths. It
    121  * will remove double separators, and unlike cwk_path_get_absolute it permits
    122  * the use of two relative paths to combine. The result will be written to a
    123  * buffer, which might be truncated if the buffer is not large enough to hold
    124  * the full path. However, the truncated result will always be
    125  * null-terminated. The returned value is the amount of characters which the
    126  * resulting path would take if it was not truncated (excluding the
    127  * null-terminating character).
    128  *
    129  * @param path_a The first path which comes first.
    130  * @param path_b The second path which comes after the first.
    131  * @param buffer The buffer where the result will be written to.
    132  * @param buffer_size The size of the result buffer.
    133  * @return Returns the total amount of characters of the full, combined path.
    134  */
    135 CWK_PUBLIC size_t cwk_path_join(const char *path_a, const char *path_b,
    136   char *buffer, size_t buffer_size);
    137 
    138 /**
    139  * @brief Joins multiple paths together.
    140  *
    141  * This function generates a new path by joining multiple paths together. It
    142  * will remove double separators, and unlike cwk_path_get_absolute it permits
    143  * the use of multiple relative paths to combine. The last path of the
    144  * submitted string array must be set to NULL. The result will be written to a
    145  * buffer, which might be truncated if the buffer is not large enough to hold
    146  * the full path. However, the truncated result will always be
    147  * null-terminated. The returned value is the amount of characters which the
    148  * resulting path would take if it was not truncated (excluding the
    149  * null-terminating character).
    150  *
    151  * @param paths An array of paths which will be joined.
    152  * @param buffer The buffer where the result will be written to.
    153  * @param buffer_size The size of the result buffer.
    154  * @return Returns the total amount of characters of the full, combined path.
    155  */
    156 CWK_PUBLIC size_t cwk_path_join_multiple(const char **paths, char *buffer,
    157   size_t buffer_size);
    158 
    159 /**
    160  * @brief Determines the root of a path.
    161  *
    162  * This function determines the root of a path by finding its length. The
    163  * root always starts at the submitted path. If the path has no root, the
    164  * length will be set to zero.
    165  *
    166  * @param path The path which will be inspected.
    167  * @param length The output of the root length.
    168  */
    169 CWK_PUBLIC void cwk_path_get_root(const char *path, size_t *length);
    170 
    171 /**
    172  * @brief Changes the root of a path.
    173  *
    174  * This function changes the root of a path. It does not normalize the result.
    175  * The result will be written to a buffer, which might be truncated if the
    176  * buffer is not large enough to hold the full path. However, the truncated
    177  * result will always be null-terminated. The returned value is the amount of
    178  * characters which the resulting path would take if it was not truncated
    179  * (excluding the null-terminating character).
    180  *
    181  * @param path The original path which will get a new root.
    182  * @param new_root The new root which will be placed in the path.
    183  * @param buffer The output buffer where the result is written to.
    184  * @param buffer_size The size of the output buffer where the result is
    185  * written to.
    186  * @return Returns the total amount of characters of the new path.
    187  */
    188 CWK_PUBLIC size_t cwk_path_change_root(const char *path, const char *new_root,
    189   char *buffer, size_t buffer_size);
    190 
    191 /**
    192  * @brief Determine whether the path is absolute or not.
    193  *
    194  * This function checks whether the path is an absolute path or not. A path is
    195  * considered to be absolute if the root ends with a separator.
    196  *
    197  * @param path The path which will be checked.
    198  * @return Returns true if the path is absolute or false otherwise.
    199  */
    200 CWK_PUBLIC bool cwk_path_is_absolute(const char *path);
    201 
    202 /**
    203  * @brief Determine whether the path is relative or not.
    204  *
    205  * This function checks whether the path is a relative path or not. A path is
    206  * considered to be relative if the root does not end with a separator.
    207  *
    208  * @param path The path which will be checked.
    209  * @return Returns true if the path is relative or false otherwise.
    210  */
    211 CWK_PUBLIC bool cwk_path_is_relative(const char *path);
    212 
    213 /**
    214  * @brief Gets the basename of a file path.
    215  *
    216  * This function gets the basename of a file path. A pointer to the beginning
    217  * of the basename will be returned through the basename parameter. This
    218  * pointer will be positioned on the first letter after the separator. The
    219  * length of the file path will be returned through the length parameter. The
    220  * length will be set to zero and the basename to NULL if there is no basename
    221  * available.
    222  *
    223  * @param path The path which will be inspected.
    224  * @param basename The output of the basename pointer.
    225  * @param length The output of the length of the basename. This may be
    226  * null if not required.
    227  */
    228 CWK_PUBLIC void cwk_path_get_basename(const char *path, const char **basename,
    229   size_t *length);
    230 
    231 /**
    232  * @brief Changes the basename of a file path.
    233  *
    234  * This function changes the basename of a file path. This function will not
    235  * write out more than the specified buffer can contain. However, the
    236  * generated string is always null-terminated - even if not the whole path is
    237  * written out. The function returns the total number of characters the
    238  * complete buffer would have, even if it was not written out completely. The
    239  * path may be the same memory address as the buffer.
    240  *
    241  * @param path The original path which will be used for the modified path.
    242  * @param new_basename The new basename which will replace the old one.
    243  * @param buffer The buffer where the changed path will be written to.
    244  * @param buffer_size The size of the result buffer where the changed path is
    245  * written to.
    246  * @return Returns the size which the complete new path would have if it was
    247  * not truncated.
    248  */
    249 CWK_PUBLIC size_t cwk_path_change_basename(const char *path,
    250   const char *new_basename, char *buffer, size_t buffer_size);
    251 
    252 /**
    253  * @brief Gets the dirname of a file path.
    254  *
    255  * This function determines the dirname of a file path and returns the length
    256  * up to which character is considered to be part of it. If no dirname is
    257  * found, the length will be set to zero. The beginning of the dirname is
    258  * always equal to the submitted path pointer.
    259  *
    260  * @param path The path which will be inspected.
    261  * @param length The length of the dirname.
    262  */
    263 CWK_PUBLIC void cwk_path_get_dirname(const char *path, size_t *length);
    264 
    265 /**
    266  * @brief Gets the extension of a file path.
    267  *
    268  * This function extracts the extension portion of a file path. A pointer to
    269  * the beginning of the extension will be returned through the extension
    270  * parameter if an extension is found and true is returned. This pointer will
    271  * be positioned on the dot. The length of the extension name will be returned
    272  * through the length parameter. If no extension is found both parameters
    273  * won't be touched and false will be returned.
    274  *
    275  * @param path The path which will be inspected.
    276  * @param extension The output of the extension pointer.
    277  * @param length The output of the length of the extension.
    278  * @return Returns true if an extension is found or false otherwise.
    279  */
    280 CWK_PUBLIC bool cwk_path_get_extension(const char *path, const char **extension,
    281   size_t *length);
    282 
    283 /**
    284  * @brief Determines whether the file path has an extension.
    285  *
    286  * This function determines whether the submitted file path has an extension.
    287  * This will evaluate to true if the last segment of the path contains a dot.
    288  *
    289  * @param path The path which will be inspected.
    290  * @return Returns true if the path has an extension or false otherwise.
    291  */
    292 CWK_PUBLIC bool cwk_path_has_extension(const char *path);
    293 
    294 /**
    295  * @brief Changes the extension of a file path.
    296  *
    297  * This function changes the extension of a file name. The function will
    298  * append an extension if the basename does not have an extension, or use the
    299  * extension as a basename if the path does not have a basename. This function
    300  * will not write out more than the specified buffer can contain. However, the
    301  * generated string is always null-terminated - even if not the whole path is
    302  * written out. The function returns the total number of characters the
    303  * complete buffer would have, even if it was not written out completely. The
    304  * path may be the same memory address as the buffer.
    305  *
    306  * @param path The path which will be used to make the change.
    307  * @param new_extension The extension which will be placed within the new
    308  * path.
    309  * @param buffer The output buffer where the result will be written to.
    310  * @param buffer_size The size of the output buffer where the result will be
    311  * written to.
    312  * @return Returns the total size which the output would have if it was not
    313  * truncated.
    314  */
    315 CWK_PUBLIC size_t cwk_path_change_extension(const char *path,
    316   const char *new_extension, char *buffer, size_t buffer_size);
    317 
    318 /**
    319  * @brief Creates a normalized version of the path.
    320  *
    321  * This function creates a normalized version of the path within the specified
    322  * buffer. This function will not write out more than the specified buffer can
    323  * contain. However, the generated string is always null-terminated - even if
    324  * not the whole path is written out. The function returns the total number of
    325  * characters the complete buffer would have, even if it was not written out
    326  * completely. The path may be the same memory address as the buffer.
    327  *
    328  * The following will be true for the normalized path:
    329  * 1) "../" will be resolved.
    330  * 2) "./" will be removed.
    331  * 3) double separators will be fixed with a single separator.
    332  * 4) separator suffixes will be removed.
    333  *
    334  * @param path The path which will be normalized.
    335  * @param buffer The buffer where the new path is written to.
    336  * @param buffer_size The size of the buffer.
    337  * @return The size which the complete normalized path has if it was not
    338  * truncated.
    339  */
    340 CWK_PUBLIC size_t cwk_path_normalize(const char *path, char *buffer,
    341   size_t buffer_size);
    342 
    343 /**
    344  * @brief Finds common portions in two paths.
    345  *
    346  * This function finds common portions in two paths and returns the number
    347  * characters from the beginning of the base path which are equal to the other
    348  * path.
    349  *
    350  * @param path_base The base path which will be compared with the other path.
    351  * @param path_other The other path which will compared with the base path.
    352  * @return Returns the number of characters which are common in the base path.
    353  */
    354 CWK_PUBLIC size_t cwk_path_get_intersection(const char *path_base,
    355   const char *path_other);
    356 
    357 /**
    358  * @brief Gets the first segment of a path.
    359  *
    360  * This function finds the first segment of a path. The position of the
    361  * segment is set to the first character after the separator, and the length
    362  * counts all characters until the next separator (excluding the separator).
    363  *
    364  * @param path The path which will be inspected.
    365  * @param segment The segment which will be extracted.
    366  * @return Returns true if there is a segment or false if there is none.
    367  */
    368 CWK_PUBLIC bool cwk_path_get_first_segment(const char *path,
    369   struct cwk_segment *segment);
    370 
    371 /**
    372  * @brief Gets the last segment of the path.
    373  *
    374  * This function gets the last segment of a path. This function may return
    375  * false if the path doesn't contain any segments, in which case the submitted
    376  * segment parameter is not modified. The position of the segment is set to
    377  * the first character after the separator, and the length counts all
    378  * characters until the end of the path (excluding the separator).
    379  *
    380  * @param path The path which will be inspected.
    381  * @param segment The segment which will be extracted.
    382  * @return Returns true if there is a segment or false if there is none.
    383  */
    384 CWK_PUBLIC bool cwk_path_get_last_segment(const char *path,
    385   struct cwk_segment *segment);
    386 
    387 /**
    388  * @brief Advances to the next segment.
    389  *
    390  * This function advances the current segment to the next segment. If there
    391  * are no more segments left, the submitted segment structure will stay
    392  * unchanged and false is returned.
    393  *
    394  * @param segment The current segment which will be advanced to the next one.
    395  * @return Returns true if another segment was found or false otherwise.
    396  */
    397 CWK_PUBLIC bool cwk_path_get_next_segment(struct cwk_segment *segment);
    398 
    399 /**
    400  * @brief Moves to the previous segment.
    401  *
    402  * This function moves the current segment to the previous segment. If the
    403  * current segment is the first one, the submitted segment structure will stay
    404  * unchanged and false is returned.
    405  *
    406  * @param segment The current segment which will be moved to the previous one.
    407  * @return Returns true if there is a segment before this one or false
    408  * otherwise.
    409  */
    410 CWK_PUBLIC bool cwk_path_get_previous_segment(struct cwk_segment *segment);
    411 
    412 /**
    413  * @brief Gets the type of the submitted path segment.
    414  *
    415  * This function inspects the contents of the segment and determines the type
    416  * of it. Currently, there are three types CWK_NORMAL, CWK_CURRENT and
    417  * CWK_BACK. A CWK_NORMAL segment is a normal folder or file entry. A
    418  * CWK_CURRENT is a "./" and a CWK_BACK a "../" segment.
    419  *
    420  * @param segment The segment which will be inspected.
    421  * @return Returns the type of the segment.
    422  */
    423 CWK_PUBLIC enum cwk_segment_type cwk_path_get_segment_type(
    424   const struct cwk_segment *segment);
    425 
    426 /**
    427  * @brief Changes the content of a segment.
    428  *
    429  * This function overrides the content of a segment to the submitted value and
    430  * outputs the whole new path to the submitted buffer. The result might
    431  * require less or more space than before if the new value length differs from
    432  * the original length. The output is truncated if the new path is larger than
    433  * the submitted buffer size, but it is always null-terminated. The source of
    434  * the segment and the submitted buffer may be the same.
    435  *
    436  * @param segment The segment which will be modifier.
    437  * @param value The new content of the segment.
    438  * @param buffer The buffer where the modified path will be written to.
    439  * @param buffer_size The size of the output buffer.
    440  * @return Returns the total size which would have been written if the output
    441  * was not truncated.
    442  */
    443 CWK_PUBLIC size_t cwk_path_change_segment(struct cwk_segment *segment,
    444   const char *value, char *buffer, size_t buffer_size);
    445 
    446 /**
    447  * @brief Checks whether the submitted pointer points to a separator.
    448  *
    449  * This function simply checks whether the submitted pointer points to a
    450  * separator, which has to be null-terminated (but not necessarily after the
    451  * separator). The function will return true if it is a separator, or false
    452  * otherwise.
    453  *
    454  * @param symbol A pointer to a string.
    455  * @return Returns true if it is a separator, or false otherwise.
    456  */
    457 CWK_PUBLIC bool cwk_path_is_separator(const char *str);
    458 
    459 /**
    460  * @brief Guesses the path style.
    461  *
    462  * This function guesses the path style based on a submitted path-string. The
    463  * guessing will look at the root and the type of slashes contained in the
    464  * path and return the style which is more likely used in the path.
    465  *
    466  * @param path The path which will be inspected.
    467  * @return Returns the style which is most likely used for the path.
    468  */
    469 CWK_PUBLIC enum cwk_path_style cwk_path_guess_style(const char *path);
    470 
    471 /**
    472  * @brief Configures which path style is used.
    473  *
    474  * This function configures which path style is used. The following styles are
    475  * currently supported.
    476  *
    477  * CWK_STYLE_WINDOWS: Use backslashes as a separator and volume for the root.
    478  * CWK_STYLE_UNIX: Use slashes as a separator and a slash for the root.
    479  *
    480  * @param style The style which will be used from now on.
    481  */
    482 CWK_PUBLIC void cwk_path_set_style(enum cwk_path_style style);
    483 
    484 /**
    485  * @brief Gets the path style configuration.
    486  *
    487  * This function gets the style configuration which is currently used for the
    488  * paths. This configuration determines how paths are parsed and generated.
    489  *
    490  * @return Returns the current path style configuration.
    491  */
    492 CWK_PUBLIC enum cwk_path_style cwk_path_get_style(void);
    493 
    494 #ifdef __cplusplus
    495 } // extern "C"
    496 #endif
    497 
    498 #endif