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