2022-10-27 15:16:54 -06:00
|
|
|
#include "arc/std/string.h"
|
|
|
|
|
|
|
|
|
|
#include "arc/std/errno.h"
|
2023-01-17 01:59:08 -07:00
|
|
|
#include <stdint.h>
|
2022-10-27 15:16:54 -06:00
|
|
|
#include <string.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
void ARC_String_Create(ARC_String **string, char *data, uint64_t length){
|
|
|
|
|
*string = (ARC_String *)malloc(sizeof(ARC_String));
|
|
|
|
|
(*string)->data = (char *)malloc(length + 1);
|
|
|
|
|
(*string)->length = length;
|
|
|
|
|
|
|
|
|
|
strncpy((*string)->data, data, length);
|
|
|
|
|
(*string)->data[length] = '\0';
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-20 22:38:29 -07:00
|
|
|
void ARC_String_CreateWithStrlen(ARC_String **string, char *data){
|
|
|
|
|
*string = (ARC_String *)malloc(sizeof(ARC_String));
|
|
|
|
|
(*string)->length = strlen(data);
|
|
|
|
|
(*string)->data = (char *)malloc((*string)->length + 1);
|
|
|
|
|
|
|
|
|
|
strncpy((*string)->data, data, (*string)->length);
|
|
|
|
|
(*string)->data[(*string)->length] = '\0';
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
void ARC_String_Destroy(ARC_String *string){
|
|
|
|
|
free(string->data);
|
|
|
|
|
free(string);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ARC_String_Copy(ARC_String **copy, ARC_String *original){
|
|
|
|
|
if(!original){
|
|
|
|
|
arc_errno = ARC_ERRNO_NULL;
|
|
|
|
|
*copy = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ARC_String_Create(copy, original->data, original->length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ARC_String_CopySubstring(ARC_String **substring, ARC_String *original, uint64_t start, uint64_t length){
|
|
|
|
|
if(!original){
|
|
|
|
|
arc_errno = ARC_ERRNO_NULL;
|
|
|
|
|
*substring = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(length == 0){
|
|
|
|
|
*substring = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(start + length > original->length){
|
|
|
|
|
arc_errno = ARC_ERRNO_DATA;
|
|
|
|
|
*substring = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char data[length];
|
|
|
|
|
for(uint32_t i = 0; i < length; i++){
|
|
|
|
|
data[i] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
strncpy(data, original->data + start, length);
|
|
|
|
|
|
|
|
|
|
ARC_String_Create(substring, data, length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ARC_String_RemoveSubstring(ARC_String **newString, ARC_String *original, ARC_String *substring){
|
|
|
|
|
uint64_t index = ARC_String_Find(original, substring);
|
|
|
|
|
if(arc_errno){
|
|
|
|
|
newString = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ARC_String_RemoveSection(newString, original, index, original->length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t ARC_String_Equals(ARC_String *first, ARC_String *second){
|
|
|
|
|
if(first->length != second->length){
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(strncmp(first->data, second->data, first->length)){
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t ARC_String_EqualsCString(ARC_String *string, const char *cstring, uint64_t length){
|
|
|
|
|
if(string->length != length){
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(strncmp(string->data, cstring, string->length)){
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-15 02:14:35 -07:00
|
|
|
uint8_t ARC_String_SubstringEqualsCString(ARC_String *string, uint64_t offset, const char *cstring, uint64_t length){
|
|
|
|
|
if(string->length - offset < length){
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(strncmp(string->data + offset, cstring, length)){
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint8_t ARC_String_Alpha(ARC_String *string){
|
|
|
|
|
for(uint64_t length = string->length; length; length--){
|
|
|
|
|
if(string->data[length - 1] >= 'a' && string->data[length - 1] <= 'z'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(string->data[length - 1] >= 'A' && string->data[length - 1] <= 'Z'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-27 15:16:54 -06:00
|
|
|
return 1;
|
|
|
|
|
}
|
2023-01-17 01:59:08 -07:00
|
|
|
|
2022-10-27 15:16:54 -06:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t ARC_String_ToUint64_t(ARC_String *string){
|
2023-01-20 22:38:29 -07:00
|
|
|
return (uint64_t) strtoul(string->data, NULL, 10);
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-20 22:38:29 -07:00
|
|
|
int64_t ARC_String_ToInt64_t(ARC_String *string){
|
|
|
|
|
return (int64_t) strtol(string->data, NULL, 10);
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
|
2023-01-20 22:38:29 -07:00
|
|
|
double ARC_String_ToDouble(ARC_String *string){
|
|
|
|
|
return strtod(string->data, NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t ARC_String_Find(ARC_String *string, ARC_String *substring){
|
|
|
|
|
if(!string || !substring){
|
2024-01-15 02:14:35 -07:00
|
|
|
ARC_DEBUG_ERR("ARC_String_Find(string, substring), string or substring was null");
|
2023-01-17 01:59:08 -07:00
|
|
|
arc_errno = ARC_ERRNO_NULL;
|
|
|
|
|
return ~(uint64_t)0;
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
if(substring->length > string->length){
|
|
|
|
|
return ~(uint64_t)0;
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t max = string->length - (substring->length - 1);
|
2022-10-27 15:16:54 -06:00
|
|
|
for(uint64_t i = 0; max; i++, max--){
|
2023-01-17 01:59:08 -07:00
|
|
|
if(!strncmp(string->data + i, substring->data, substring->length)){
|
2023-01-20 22:38:29 -07:00
|
|
|
return i;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
return ~(uint64_t)0;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t ARC_String_FindCString(ARC_String *string, const char *cstring, uint64_t length){
|
|
|
|
|
if(!string || !cstring){
|
|
|
|
|
arc_errno = ARC_ERRNO_NULL;
|
2024-01-15 02:14:35 -07:00
|
|
|
ARC_DEBUG_ERR("ARC_String_FindCString(string, cstring, length), string or cstring was null");
|
2023-01-17 01:59:08 -07:00
|
|
|
return ~(uint64_t)0;
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
if(string->length < length){
|
|
|
|
|
return ~(uint64_t)0;
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t max = string->length - (length - 1);
|
|
|
|
|
for(uint64_t i = 0; i < max; i++){
|
|
|
|
|
if(!strncmp(string->data + i, cstring, length)){
|
|
|
|
|
return i + 1;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
return ~(uint64_t)0;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t ARC_String_FindBack(ARC_String *string, ARC_String *substring){
|
|
|
|
|
if(!string || !substring){
|
|
|
|
|
arc_errno = ARC_ERRNO_NULL;
|
2024-01-15 02:14:35 -07:00
|
|
|
ARC_DEBUG_ERR("ARC_String_FindBack(string, substring), string or substring was null");
|
2023-01-17 01:59:08 -07:00
|
|
|
return ~(uint64_t)0;
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
if(substring->length > string->length){
|
|
|
|
|
return ~(uint64_t)0;
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t max = string->length - (substring->length - 1);
|
2022-10-27 15:16:54 -06:00
|
|
|
for(; max; max--){
|
2023-01-17 01:59:08 -07:00
|
|
|
if(!strncmp(string->data + (max - 1), substring->data, substring->length)){
|
|
|
|
|
return max;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ~(uint64_t)0;
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-16 23:58:31 -07:00
|
|
|
void ARC_String_StripEnds(ARC_String **stripped, ARC_String *original, char charToStrip){
|
2023-01-17 01:59:08 -07:00
|
|
|
if(!original){
|
|
|
|
|
arc_errno = ARC_ERRNO_NULL;
|
|
|
|
|
*stripped = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(!original->length){
|
|
|
|
|
arc_errno = ARC_ERRNO_DATA;
|
|
|
|
|
*stripped = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint64_t length = original->length - 1;
|
|
|
|
|
for(; length; length--){
|
|
|
|
|
if(strncmp(original->data + (length - 1), &charToStrip, 1)){
|
2022-10-27 15:16:54 -06:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
if(!length){
|
|
|
|
|
arc_errno = ARC_ERRNO_DATA;
|
|
|
|
|
*stripped = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint64_t start = 0;
|
|
|
|
|
for(; start <= length; start++){
|
|
|
|
|
if(strncmp(original->data + start, &charToStrip, 1)){
|
|
|
|
|
break;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
if(start == length){
|
|
|
|
|
arc_errno = ARC_ERRNO_DATA;
|
|
|
|
|
*stripped = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
length -= start;
|
|
|
|
|
ARC_String_Create(stripped, original->data + start, length);
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
|
2024-01-16 23:58:31 -07:00
|
|
|
void ARC_String_StripWhitespace(ARC_String **stripped, ARC_String *original){
|
2023-01-17 01:59:08 -07:00
|
|
|
if(!original){
|
|
|
|
|
arc_errno = ARC_ERRNO_NULL;
|
|
|
|
|
*stripped = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
if(!original->length){
|
|
|
|
|
arc_errno = ARC_ERRNO_DATA;
|
|
|
|
|
*stripped = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
2022-10-27 15:16:54 -06:00
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t length = 0;
|
|
|
|
|
for(uint64_t i = 0; i < original->length; i++){
|
|
|
|
|
if(original->data[i] == ' '){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i] == '\n'){
|
|
|
|
|
continue;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
2023-01-17 01:59:08 -07:00
|
|
|
|
|
|
|
|
if(original->data[i] == '\t'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i] == '\r'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
length++;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
if(!length){
|
|
|
|
|
arc_errno = ARC_ERRNO_DATA;
|
|
|
|
|
*stripped = NULL;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
length++;
|
|
|
|
|
char data[length];
|
|
|
|
|
for(uint32_t i = 0; i < length; i++){
|
|
|
|
|
data[i] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint64_t start = 0;
|
|
|
|
|
for(uint64_t i = 0; i < length; i++){
|
|
|
|
|
if(original->data[i] == ' '){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i] == '\n'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i] == '\t'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i] == '\r'){
|
|
|
|
|
continue;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
2023-01-17 01:59:08 -07:00
|
|
|
|
|
|
|
|
data[start] = original->data[i];
|
|
|
|
|
start++;
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
|
|
|
|
|
2023-01-17 01:59:08 -07:00
|
|
|
ARC_String_Create(stripped, data, length);
|
2022-10-27 15:16:54 -06:00
|
|
|
}
|
2023-01-17 01:59:08 -07:00
|
|
|
|
2024-01-16 23:58:31 -07:00
|
|
|
void ARC_String_StripEndsWhitespace(ARC_String **stripped, ARC_String *original){
|
2023-01-17 01:59:08 -07:00
|
|
|
uint64_t index;
|
|
|
|
|
for(uint64_t i = 0; i < original->length; i++){
|
|
|
|
|
if(original->data[i] == ' '){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i] == '\n'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i] == '\t'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i] == '\r'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
index = i;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint64_t endIndex;
|
2023-08-22 04:08:53 +00:00
|
|
|
for(uint64_t i = original->length;; i--){
|
2023-01-17 01:59:08 -07:00
|
|
|
if(original->data[i - 1] == ' '){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i - 1] == '\n'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i - 1] == '\t'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(original->data[i - 1] == '\r'){
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
endIndex = i;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ARC_String_CopySubstring(stripped, original, index, endIndex - index);
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-16 23:58:31 -07:00
|
|
|
void ARC_String_Merge(ARC_String **combined, ARC_String *first, ARC_String *second){
|
2023-01-17 01:59:08 -07:00
|
|
|
char data[first->length + second->length];
|
|
|
|
|
for(uint32_t i = 0; i < first->length; i++){
|
|
|
|
|
data[i] = first->data[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for(uint32_t i = 0; i < second->length; i++){
|
|
|
|
|
data[i + first->length] = second->data[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ARC_String_Create(combined, data, first->length + second->length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ARC_String_RemoveSection(ARC_String **newString, ARC_String *original, uint64_t removeIndex, uint64_t removeLength){
|
|
|
|
|
if(removeIndex == 0 && removeIndex + removeLength >= original->length){
|
|
|
|
|
ARC_String_Copy(newString, original);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(removeIndex == 0){
|
|
|
|
|
ARC_String_CopySubstring(newString, original, removeLength, original->length - removeLength);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(removeIndex + removeLength >= original->length){
|
|
|
|
|
ARC_String_CopySubstring(newString, original, 0, removeIndex);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ARC_String *first, *second;
|
|
|
|
|
ARC_String_CopySubstring(&first , original, 0 , removeIndex );
|
|
|
|
|
ARC_String_CopySubstring(&second, original, removeIndex + removeLength, original->length - (removeIndex + removeLength));
|
|
|
|
|
|
2024-01-16 23:58:31 -07:00
|
|
|
ARC_String_Merge(newString, first, second);
|
2023-01-17 01:59:08 -07:00
|
|
|
|
|
|
|
|
ARC_String_Destroy(first );
|
|
|
|
|
ARC_String_Destroy(second);
|
|
|
|
|
}
|