mirror of
https://github.com/theoludwig/libcproject.git
synced 2024-12-11 21:13:00 +01:00
Compare commits
No commits in common. "develop" and "v4.2.0" have entirely different histories.
8
.github/workflows/ci.yml
vendored
8
.github/workflows/ci.yml
vendored
@ -4,13 +4,13 @@ on:
|
||||
push:
|
||||
branches: [develop]
|
||||
pull_request:
|
||||
branches: [main, develop]
|
||||
branches: [master, develop]
|
||||
|
||||
jobs:
|
||||
ci:
|
||||
runs-on: "ubuntu-latest"
|
||||
steps:
|
||||
- uses: "actions/checkout@v4.1.7"
|
||||
- uses: "actions/checkout@v4.1.1"
|
||||
|
||||
- run: "sudo apt update"
|
||||
|
||||
@ -33,8 +33,8 @@ jobs:
|
||||
lint-commit:
|
||||
runs-on: "ubuntu-latest"
|
||||
steps:
|
||||
- uses: "actions/checkout@v4.1.7"
|
||||
- uses: "actions/checkout@v4.1.1"
|
||||
with:
|
||||
fetch-depth: 0
|
||||
|
||||
- uses: "wagoid/commitlint-github-action@v6.1.2"
|
||||
- uses: "wagoid/commitlint-github-action@v5.4.4"
|
||||
|
10
.github/workflows/release.yml
vendored
10
.github/workflows/release.yml
vendored
@ -2,13 +2,13 @@ name: "Release"
|
||||
|
||||
on:
|
||||
push:
|
||||
branches: [main]
|
||||
branches: [master]
|
||||
|
||||
jobs:
|
||||
release:
|
||||
runs-on: "ubuntu-latest"
|
||||
steps:
|
||||
- uses: "actions/checkout@v4.1.7"
|
||||
- uses: "actions/checkout@v4.1.1"
|
||||
with:
|
||||
fetch-depth: 0
|
||||
persist-credentials: false
|
||||
@ -32,12 +32,12 @@ jobs:
|
||||
- run: "make set_version"
|
||||
|
||||
- name: "Use Node.js"
|
||||
uses: "actions/setup-node@v4.0.3"
|
||||
uses: "actions/setup-node@v4.0.1"
|
||||
with:
|
||||
node-version: "20.17.0"
|
||||
node-version: "20.10.0"
|
||||
|
||||
- name: "Install Release Tools"
|
||||
run: "npm install --save-dev semantic-release@23.1.1 @commitlint/cli@19.5.0 @commitlint/config-conventional@19.5.0 @semantic-release/git@10.0.1 @semantic-release/exec@6.0.3 @saithodev/semantic-release-backmerge@4.0.1 vercel@37.4.2"
|
||||
run: "npm install --save-dev semantic-release@22.0.12 @commitlint/cli@18.4.3 @commitlint/config-conventional@18.4.3 @semantic-release/git@10.0.1 @semantic-release/exec@6.0.3 @saithodev/semantic-release-backmerge@4.0.1 vercel@33.0.1"
|
||||
|
||||
- run: "rm --force package.json package-lock.json"
|
||||
|
||||
|
@ -1,5 +1,5 @@
|
||||
{
|
||||
"branches": ["main"],
|
||||
"branches": ["master"],
|
||||
"plugins": [
|
||||
[
|
||||
"@semantic-release/commit-analyzer",
|
||||
@ -30,7 +30,7 @@
|
||||
[
|
||||
"@saithodev/semantic-release-backmerge",
|
||||
{
|
||||
"branches": [{ "from": "main", "to": "develop" }],
|
||||
"branches": [{ "from": "master", "to": "develop" }],
|
||||
"backmergeStrategy": "merge"
|
||||
}
|
||||
]
|
||||
|
@ -30,18 +30,3 @@ If you're adding new features to **libcproject**, please include tests.
|
||||
## Commits
|
||||
|
||||
The commit message guidelines adheres to [Conventional Commits](https://www.conventionalcommits.org/) and [Semantic Versioning](https://semver.org/) for releases.
|
||||
|
||||
## Git Submodules
|
||||
|
||||
To get the submodule:
|
||||
|
||||
```sh
|
||||
git submodule update --init --recursive
|
||||
```
|
||||
|
||||
To update the version:
|
||||
|
||||
```sh
|
||||
cd doxygen-awesome-css
|
||||
git checkout v2.3.3
|
||||
```
|
||||
|
4
LICENSE
4
LICENSE
@ -1,6 +1,6 @@
|
||||
# MIT License
|
||||
MIT License
|
||||
|
||||
Copyright (c) Théo LUDWIG <contact@theoludwig.fr>
|
||||
Copyright (c) Théo LUDWIG
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
|
@ -1 +1 @@
|
||||
Subproject commit 40e9b25b6174dd3b472d8868f63323a870dfeeb8
|
||||
Subproject commit df88fe4fdd97714fadfd3ef17de0b4401f804052
|
@ -1,35 +1,18 @@
|
||||
#include "array_list.h"
|
||||
|
||||
struct array_list* array_list_initialization() {
|
||||
return array_list_initialization_with_capacity(ARRAY_LIST_INITIAL_CAPACITY);
|
||||
}
|
||||
|
||||
struct array_list* array_list_initialization_with_capacity(size_t capacity) {
|
||||
struct array_list* list = malloc(sizeof(struct array_list));
|
||||
if (list == NULL) {
|
||||
perror("Error (array_list_initialization)");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
list->data = malloc(sizeof(void*) * ARRAY_LIST_INITIAL_CAPACITY);
|
||||
list->size = 0;
|
||||
list->capacity = capacity;
|
||||
list->capacity_step = capacity;
|
||||
list->data = malloc(sizeof(void*) * list->capacity);
|
||||
if (list->data == NULL) {
|
||||
perror("Error (array_list_initialization)");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
list->capacity = ARRAY_LIST_INITIAL_CAPACITY;
|
||||
return list;
|
||||
}
|
||||
|
||||
void array_list_add(struct array_list* list, void* element) {
|
||||
if (list->size >= list->capacity) {
|
||||
size_t previous_capacity = list->capacity;
|
||||
list->capacity += list->capacity_step;
|
||||
list->capacity += ARRAY_LIST_INITIAL_CAPACITY;
|
||||
list->data = realloc(list->data, sizeof(void*) * list->capacity);
|
||||
if (list->data == NULL) {
|
||||
perror("Error (array_list_add)");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
for (size_t index = previous_capacity; index < list->capacity; index++) {
|
||||
list->data[index] = NULL;
|
||||
}
|
||||
|
@ -3,7 +3,6 @@
|
||||
|
||||
#include <errno.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "types.h"
|
||||
@ -18,7 +17,6 @@ struct array_list {
|
||||
void** data;
|
||||
size_t size;
|
||||
size_t capacity;
|
||||
size_t capacity_step;
|
||||
};
|
||||
|
||||
/**
|
||||
@ -27,12 +25,6 @@ struct array_list {
|
||||
*/
|
||||
struct array_list* array_list_initialization();
|
||||
|
||||
/**
|
||||
* @brief Initializes a new array list with a capacity.
|
||||
* @since v5.0.0
|
||||
*/
|
||||
struct array_list* array_list_initialization_with_capacity(size_t capacity);
|
||||
|
||||
/**
|
||||
* @brief Adds an element to the end of the array list.
|
||||
*
|
||||
|
30
lib/assert.h
30
lib/assert.h
@ -1,30 +0,0 @@
|
||||
#ifndef __LIBCPROJECT_ASSERT__
|
||||
#define __LIBCPROJECT_ASSERT__
|
||||
|
||||
#include <assert.h>
|
||||
#include <errno.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "string.h"
|
||||
#include "types.h"
|
||||
|
||||
/**
|
||||
* @brief Check if the two strings are equal. If they are not equal, print the expected and actual strings.
|
||||
*
|
||||
* @param character
|
||||
* @return bool
|
||||
* @since v5.0.0
|
||||
*/
|
||||
bool assert_string_equal(const string_t actual, const string_t expected);
|
||||
|
||||
/**
|
||||
* @brief Check if the two strings are not equal. If they are equal, print the expected and actual strings.
|
||||
*
|
||||
* @param character
|
||||
* @return bool
|
||||
* @since v5.0.0
|
||||
*/
|
||||
bool assert_string_not_equal(const string_t actual, const string_t expected);
|
||||
|
||||
#endif
|
@ -32,9 +32,9 @@ bool character_get_is_digit(const char character) {
|
||||
return character >= '0' && character <= '9';
|
||||
}
|
||||
|
||||
uint8_t character_get_alphabet_position(const char character) {
|
||||
unsigned char character_get_alphabet_position(const char character) {
|
||||
const char letter = character_to_lower(character);
|
||||
uint8_t position = 0;
|
||||
unsigned char position = 0;
|
||||
if (letter >= 'a' && letter <= 'z') {
|
||||
position = (letter - 'a') + 1;
|
||||
}
|
||||
|
@ -48,7 +48,8 @@ char character_to_lower(const char character);
|
||||
* @brief Check if the character is a digit ('0', '1', '2', '3', '4', '5', '6', '7, '8' or '9').
|
||||
*
|
||||
* @param character
|
||||
* @return bool
|
||||
* @return true
|
||||
* @return false
|
||||
* @since v1.0.0
|
||||
*/
|
||||
bool character_get_is_digit(const char character);
|
||||
@ -58,9 +59,9 @@ bool character_get_is_digit(const char character);
|
||||
* Return 0 if the character is not a letter.
|
||||
*
|
||||
* @param character
|
||||
* @return uint8_t
|
||||
* @return unsigned char
|
||||
* @since v1.0.0
|
||||
*/
|
||||
uint8_t character_get_alphabet_position(const char character);
|
||||
unsigned char character_get_alphabet_position(const char character);
|
||||
|
||||
#endif
|
||||
|
@ -19,9 +19,9 @@ char convert_digit_to_character(const char digit) {
|
||||
return digit + '0';
|
||||
}
|
||||
|
||||
int64_t convert_string_to_number(const string_t string) {
|
||||
long long convert_string_to_number(const string_t string) {
|
||||
bool is_negative = string[0] == '-';
|
||||
int64_t integer = 0;
|
||||
long long integer = 0;
|
||||
size_t length = string_get_length(string);
|
||||
for (size_t index = is_negative ? 1 : 0; index < length; index++) {
|
||||
integer = integer * 10 + convert_character_to_digit(string[index]);
|
||||
@ -29,13 +29,13 @@ int64_t convert_string_to_number(const string_t string) {
|
||||
return is_negative ? integer * -1 : integer;
|
||||
}
|
||||
|
||||
string_t convert_number_to_string(const int64_t integer) {
|
||||
string_t convert_number_to_string(const long long integer) {
|
||||
if (integer == 0) {
|
||||
return convert_character_to_string('0');
|
||||
}
|
||||
bool is_negative = integer < 0;
|
||||
size_t length = 1;
|
||||
int64_t current = mathematics_absolute_value(integer);
|
||||
long long current = mathematics_absolute_value(integer);
|
||||
while (current != 0) {
|
||||
current = current / 10;
|
||||
length++;
|
||||
@ -62,21 +62,21 @@ string_t convert_number_to_string(const int64_t integer) {
|
||||
return string;
|
||||
}
|
||||
|
||||
string_t convert_number_from_base_10_to_base(uint64_t number, uint64_t base) {
|
||||
string_t convert_number_from_base_10_to_base(unsigned long long number, unsigned long base) {
|
||||
if (number == 0) {
|
||||
return "0";
|
||||
}
|
||||
int64_t remainders[64];
|
||||
int64_t index = 0;
|
||||
int remainders[64];
|
||||
int index = 0;
|
||||
while (number > 0) {
|
||||
remainders[index] = number % base;
|
||||
number = number / base;
|
||||
index++;
|
||||
}
|
||||
string_t result = malloc(sizeof(char) * (index + 1));
|
||||
int64_t index_result = 0;
|
||||
for (int64_t iteration = index - 1; iteration >= 0; iteration--) {
|
||||
int64_t remainder = remainders[iteration];
|
||||
int index_result = 0;
|
||||
for (int iteration = index - 1; iteration >= 0; iteration--) {
|
||||
int remainder = remainders[iteration];
|
||||
if (remainder >= 10) {
|
||||
result[index_result] = (char)((remainder - 10) + 'A');
|
||||
} else {
|
||||
@ -88,17 +88,17 @@ string_t convert_number_from_base_10_to_base(uint64_t number, uint64_t base) {
|
||||
return result;
|
||||
}
|
||||
|
||||
uint64_t convert_number_from_base_to_base_10(string_t number, uint64_t base) {
|
||||
unsigned long convert_number_from_base_to_base_10(string_t number, unsigned long base) {
|
||||
size_t length = string_get_length(number);
|
||||
int64_t exponent = length - 1;
|
||||
uint64_t result = 0;
|
||||
int64_t index = 0;
|
||||
int exponent = length - 1;
|
||||
unsigned long result = 0;
|
||||
int index = 0;
|
||||
while (exponent >= 0) {
|
||||
int64_t current_number = (int64_t)(number[index] - '0');
|
||||
int current_number = (int)(number[index] - '0');
|
||||
if (current_number >= 10) {
|
||||
current_number = (int64_t)(number[index] - 'A') + 10;
|
||||
current_number = (int)(number[index] - 'A') + 10;
|
||||
} else {
|
||||
current_number = (int64_t)(number[index] - '0');
|
||||
current_number = (int)(number[index] - '0');
|
||||
}
|
||||
result = result + current_number * mathematics_pow(base, exponent);
|
||||
exponent--;
|
||||
@ -107,6 +107,6 @@ uint64_t convert_number_from_base_to_base_10(string_t number, uint64_t base) {
|
||||
return result;
|
||||
}
|
||||
|
||||
string_t convert_number_from_base_to_another(string_t number, uint64_t base_from, uint64_t base_target) {
|
||||
string_t convert_number_from_base_to_another(string_t number, unsigned long base_from, unsigned long base_target) {
|
||||
return convert_number_from_base_10_to_base(convert_number_from_base_to_base_10(number, base_from), base_target);
|
||||
}
|
||||
|
@ -42,10 +42,10 @@ char convert_digit_to_character(const char digit);
|
||||
* @brief Convert a string to a number.
|
||||
*
|
||||
* @param string
|
||||
* @return int64_t
|
||||
* @return long long
|
||||
* @since v1.0.0
|
||||
*/
|
||||
int64_t convert_string_to_number(const string_t string);
|
||||
long long convert_string_to_number(const string_t string);
|
||||
|
||||
/**
|
||||
* @brief Convert a number to a string.
|
||||
@ -54,7 +54,7 @@ int64_t convert_string_to_number(const string_t string);
|
||||
* @return string_t
|
||||
* @since v1.0.0
|
||||
*/
|
||||
string_t convert_number_to_string(const int64_t integer);
|
||||
string_t convert_number_to_string(const long long integer);
|
||||
|
||||
/**
|
||||
* @brief Convert a number (base 10) to a string with a specific base.
|
||||
@ -64,17 +64,17 @@ string_t convert_number_to_string(const int64_t integer);
|
||||
* @return string_t
|
||||
* @since v1.0.0
|
||||
*/
|
||||
string_t convert_number_from_base_10_to_base(uint64_t number, uint64_t base);
|
||||
string_t convert_number_from_base_10_to_base(unsigned long long number, unsigned long base);
|
||||
|
||||
/**
|
||||
* @brief Convert a number with a specific base to a number base 10.
|
||||
*
|
||||
* @param number
|
||||
* @param base
|
||||
* @return uint64_t
|
||||
* @return int
|
||||
* @since v1.0.0
|
||||
*/
|
||||
uint64_t convert_number_from_base_to_base_10(string_t number, uint64_t base);
|
||||
unsigned long convert_number_from_base_to_base_10(string_t number, unsigned long base);
|
||||
|
||||
/**
|
||||
* @brief Convert a number with a specific base to a number of specific base.
|
||||
@ -85,6 +85,6 @@ uint64_t convert_number_from_base_to_base_10(string_t number, uint64_t base);
|
||||
* @return string_t
|
||||
* @since v1.0.0
|
||||
*/
|
||||
string_t convert_number_from_base_to_another(string_t number, uint64_t base_from, uint64_t base_target);
|
||||
string_t convert_number_from_base_to_another(string_t number, unsigned long base_from, unsigned long base_target);
|
||||
|
||||
#endif
|
||||
|
339
lib/date.c
339
lib/date.c
@ -1,339 +0,0 @@
|
||||
#include "date.h"
|
||||
|
||||
struct date *date_copy(struct date *date_to_copy) {
|
||||
struct date *date = malloc(sizeof(struct date));
|
||||
if (date == NULL) {
|
||||
perror("Error (date_copy)");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
date->year = date_to_copy->year;
|
||||
date->month = date_to_copy->month;
|
||||
date->day = date_to_copy->day;
|
||||
date->hours = date_to_copy->hours;
|
||||
date->minutes = date_to_copy->minutes;
|
||||
date->seconds = date_to_copy->seconds;
|
||||
date->milliseconds = date_to_copy->milliseconds;
|
||||
date->timezone_utc_offset = date_to_copy->timezone_utc_offset;
|
||||
return date;
|
||||
}
|
||||
|
||||
bool date_get_is_valid_year(uint16_t year) {
|
||||
return year <= 9999;
|
||||
}
|
||||
|
||||
bool date_get_is_valid_month(uint8_t month) {
|
||||
return month >= 1 && month <= 12;
|
||||
}
|
||||
|
||||
bool date_get_is_valid_day(uint8_t day) {
|
||||
return day >= 1 && day <= 31;
|
||||
}
|
||||
|
||||
bool date_get_is_valid_hours(uint8_t hours) {
|
||||
return hours <= 23;
|
||||
}
|
||||
|
||||
bool date_get_is_valid_minutes(uint8_t minutes) {
|
||||
return minutes <= 59;
|
||||
}
|
||||
|
||||
bool date_get_is_valid_seconds(uint8_t seconds) {
|
||||
return seconds <= 59;
|
||||
}
|
||||
|
||||
bool date_get_is_valid_milliseconds(uint16_t milliseconds) {
|
||||
return milliseconds <= 999;
|
||||
}
|
||||
|
||||
bool date_get_is_valid_timezone_utc_offset(int8_t timezone_utc_offset) {
|
||||
return timezone_utc_offset >= -12 && timezone_utc_offset <= 14;
|
||||
}
|
||||
|
||||
bool date_get_is_valid(struct date *date) {
|
||||
size_t date_days_of_month = date_get_days_of_month(date->month, date->year);
|
||||
return (date_get_is_valid_month(date->month) &&
|
||||
date_get_is_valid_day(date->day) &&
|
||||
date->day <= date_days_of_month &&
|
||||
date_get_is_valid_hours(date->hours) &&
|
||||
date_get_is_valid_minutes(date->minutes) &&
|
||||
date_get_is_valid_seconds(date->seconds) &&
|
||||
date_get_is_valid_milliseconds(date->milliseconds) &&
|
||||
date_get_is_valid_timezone_utc_offset(date->timezone_utc_offset));
|
||||
}
|
||||
|
||||
string_t date_to_iso_string(struct date *date_original) {
|
||||
struct date *date = date_copy(date_original);
|
||||
date_to_utc(date);
|
||||
|
||||
size_t iso_string_length = 24;
|
||||
string_t result = malloc(sizeof(char) * (iso_string_length + 1));
|
||||
|
||||
string_t year_string = string_zero_pad(date->year, 4);
|
||||
string_t month_string = string_zero_pad(date->month, 2);
|
||||
string_t day_string = string_zero_pad(date->day, 2);
|
||||
string_t hours_string = string_zero_pad(date->hours, 2);
|
||||
string_t minutes_string = string_zero_pad(date->minutes, 2);
|
||||
string_t seconds_string = string_zero_pad(date->seconds, 2);
|
||||
string_t milliseconds_string = string_zero_pad(date->milliseconds, 3);
|
||||
sprintf(result, "%s-%s-%sT%s:%s:%s.%sZ", year_string, month_string, day_string, hours_string, minutes_string, seconds_string, milliseconds_string);
|
||||
free(year_string);
|
||||
free(month_string);
|
||||
free(day_string);
|
||||
free(hours_string);
|
||||
free(minutes_string);
|
||||
free(seconds_string);
|
||||
free(milliseconds_string);
|
||||
|
||||
free(date);
|
||||
return result;
|
||||
}
|
||||
|
||||
string_t date_to_iso_string_without_time(struct date *date) {
|
||||
size_t iso_string_length = 10;
|
||||
string_t result = malloc(sizeof(char) * (iso_string_length + 1));
|
||||
if (result == NULL) {
|
||||
perror("Error (date_to_iso_string_without_time)");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
string_t year_string = string_zero_pad(date->year, 4);
|
||||
string_t month_string = string_zero_pad(date->month, 2);
|
||||
string_t day_string = string_zero_pad(date->day, 2);
|
||||
sprintf(result, "%s-%s-%s", year_string, month_string, day_string);
|
||||
free(year_string);
|
||||
free(month_string);
|
||||
free(day_string);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
struct date *date_from_iso_string(string_t iso_string) {
|
||||
struct date *date = malloc(sizeof(struct date));
|
||||
if (date == NULL) {
|
||||
perror("Error (date_from_iso_string)");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
string_t year_string = string_substring(iso_string, 0, 3);
|
||||
date->year = (uint16_t)convert_string_to_number(year_string);
|
||||
free(year_string);
|
||||
|
||||
string_t month_string = string_substring(iso_string, 5, 6);
|
||||
date->month = (uint8_t)convert_string_to_number(month_string);
|
||||
free(month_string);
|
||||
|
||||
string_t day_string = string_substring(iso_string, 8, 9);
|
||||
date->day = (uint8_t)convert_string_to_number(day_string);
|
||||
free(day_string);
|
||||
|
||||
string_t hours_string = string_substring(iso_string, 11, 12);
|
||||
date->hours = (uint8_t)convert_string_to_number(hours_string);
|
||||
free(hours_string);
|
||||
|
||||
string_t minutes_string = string_substring(iso_string, 14, 15);
|
||||
date->minutes = (uint8_t)convert_string_to_number(minutes_string);
|
||||
free(minutes_string);
|
||||
|
||||
string_t seconds_string = string_substring(iso_string, 17, 18);
|
||||
date->seconds = (uint8_t)convert_string_to_number(seconds_string);
|
||||
free(seconds_string);
|
||||
|
||||
string_t milliseconds_string = string_substring(iso_string, 20, 22);
|
||||
date->milliseconds = (uint16_t)convert_string_to_number(milliseconds_string);
|
||||
free(milliseconds_string);
|
||||
|
||||
date->timezone_utc_offset = 0;
|
||||
|
||||
return date;
|
||||
}
|
||||
|
||||
uint8_t date_get_days_of_month(uint8_t month, uint16_t year) {
|
||||
switch (month) {
|
||||
case 1:
|
||||
return 31;
|
||||
case 2:
|
||||
return date_get_is_leap_year(year) ? 29 : 28;
|
||||
case 3:
|
||||
return 31;
|
||||
case 4:
|
||||
return 30;
|
||||
case 5:
|
||||
return 31;
|
||||
case 6:
|
||||
return 30;
|
||||
case 7:
|
||||
return 31;
|
||||
case 8:
|
||||
return 31;
|
||||
case 9:
|
||||
return 30;
|
||||
case 10:
|
||||
return 31;
|
||||
case 11:
|
||||
return 30;
|
||||
case 12:
|
||||
return 31;
|
||||
default:
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
bool date_get_is_leap_year(uint16_t year) {
|
||||
return (year % 4 == 0 && year % 100 != 0) || year % 400 == 0;
|
||||
}
|
||||
|
||||
uint64_t date_convert_milliseconds_to_seconds(uint16_t milliseconds) {
|
||||
return milliseconds / MILLISECONDS_PER_SECOND;
|
||||
}
|
||||
|
||||
uint64_t date_convert_seconds_to_milliseconds(uint64_t seconds) {
|
||||
return seconds * MILLISECONDS_PER_SECOND;
|
||||
}
|
||||
|
||||
uint64_t date_convert_days_to_seconds(uint64_t days) {
|
||||
return days * SECONDS_PER_DAY;
|
||||
}
|
||||
|
||||
uint64_t date_convert_hms_to_seconds(uint8_t hours, uint8_t minutes, uint8_t seconds) {
|
||||
return (hours * SECONDS_PER_HOUR) + (minutes * SECONDS_PER_MINUTE) + seconds;
|
||||
}
|
||||
|
||||
uint64_t date_to_total_seconds(struct date *date) {
|
||||
uint64_t total_seconds = 0;
|
||||
|
||||
for (uint16_t year = 0; year < date->year; year++) {
|
||||
total_seconds += 365 * SECONDS_PER_DAY;
|
||||
if (date_get_is_leap_year(year)) {
|
||||
total_seconds += SECONDS_PER_DAY;
|
||||
}
|
||||
}
|
||||
|
||||
for (uint8_t month = 1; month < date->month; month++) {
|
||||
total_seconds += date_convert_days_to_seconds(date_get_days_of_month(month, date->year));
|
||||
}
|
||||
|
||||
total_seconds += date_convert_days_to_seconds(date->day - 1);
|
||||
total_seconds += date_convert_hms_to_seconds(date->hours, date->minutes, date->seconds);
|
||||
|
||||
return total_seconds;
|
||||
}
|
||||
|
||||
uint64_t date_duration_seconds_between_2_dates(struct date *date1, struct date *date2) {
|
||||
struct date *utc_date1 = date_copy(date1);
|
||||
struct date *utc_date2 = date_copy(date2);
|
||||
date_to_utc(utc_date1);
|
||||
date_to_utc(utc_date2);
|
||||
|
||||
uint64_t total_seconds_date1 = date_to_total_seconds(utc_date1);
|
||||
uint64_t total_seconds_date2 = date_to_total_seconds(utc_date2);
|
||||
|
||||
free(utc_date1);
|
||||
free(utc_date2);
|
||||
|
||||
return total_seconds_date1 > total_seconds_date2 ? total_seconds_date1 - total_seconds_date2 : total_seconds_date2 - total_seconds_date1;
|
||||
}
|
||||
|
||||
void date_add_hours(struct date *date, int64_t hours) {
|
||||
if (hours == 0) {
|
||||
return;
|
||||
}
|
||||
int64_t total_hours = date->hours + hours;
|
||||
int64_t additional_days = total_hours / 24;
|
||||
int64_t new_hours = total_hours % 24;
|
||||
|
||||
if (new_hours < 0) {
|
||||
new_hours += 24;
|
||||
additional_days -= 1;
|
||||
}
|
||||
|
||||
date->hours = (uint8_t)new_hours;
|
||||
|
||||
if (additional_days != 0) {
|
||||
date->day += additional_days;
|
||||
|
||||
while (date->day > date_get_days_of_month(date->month, date->year)) {
|
||||
date->day -= date_get_days_of_month(date->month, date->year);
|
||||
date->month++;
|
||||
|
||||
if (date->month > 12) {
|
||||
date->month = 1;
|
||||
date->year++;
|
||||
}
|
||||
}
|
||||
|
||||
while (date->day < 1) {
|
||||
date->month--;
|
||||
if (date->month < 1) {
|
||||
date->month = 12;
|
||||
date->year--;
|
||||
}
|
||||
|
||||
date->day += date_get_days_of_month(date->month, date->year);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void date_add_days(struct date *date, int64_t days) {
|
||||
date_add_hours(date, days * 24);
|
||||
}
|
||||
|
||||
void date_to_utc(struct date *date) {
|
||||
if (date->timezone_utc_offset == 0) {
|
||||
return;
|
||||
}
|
||||
int8_t timezone_utc_offset = date->timezone_utc_offset;
|
||||
date->timezone_utc_offset = 0;
|
||||
date_add_hours(date, mathematics_opposite(timezone_utc_offset));
|
||||
}
|
||||
|
||||
struct date *date_get_now_utc() {
|
||||
struct date *date = malloc(sizeof(struct date));
|
||||
if (date == NULL) {
|
||||
perror("Error (date_get_now_utc)");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
time_t current_time = time(NULL);
|
||||
struct tm *current_time_tm = gmtime(¤t_time);
|
||||
|
||||
date->year = (uint16_t)(current_time_tm->tm_year + 1900);
|
||||
date->month = (uint8_t)(current_time_tm->tm_mon + 1);
|
||||
date->day = (uint8_t)current_time_tm->tm_mday;
|
||||
date->hours = (uint8_t)current_time_tm->tm_hour;
|
||||
date->minutes = (uint8_t)current_time_tm->tm_min;
|
||||
date->seconds = (uint8_t)current_time_tm->tm_sec;
|
||||
date->milliseconds = 0;
|
||||
date->timezone_utc_offset = 0;
|
||||
|
||||
return date;
|
||||
}
|
||||
|
||||
struct date *date_get_now_local() {
|
||||
struct date *date = malloc(sizeof(struct date));
|
||||
if (date == NULL) {
|
||||
perror("Error (date_get_now_local)");
|
||||
exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
time_t current_time = time(NULL);
|
||||
struct tm *current_time_tm = localtime(¤t_time);
|
||||
|
||||
date->year = (uint16_t)(current_time_tm->tm_year + 1900);
|
||||
date->month = (uint8_t)(current_time_tm->tm_mon + 1);
|
||||
date->day = (uint8_t)current_time_tm->tm_mday;
|
||||
date->hours = (uint8_t)current_time_tm->tm_hour;
|
||||
date->minutes = (uint8_t)current_time_tm->tm_min;
|
||||
date->seconds = (uint8_t)current_time_tm->tm_sec;
|
||||
date->milliseconds = 0;
|
||||
date->timezone_utc_offset = 0;
|
||||
|
||||
return date;
|
||||
}
|
||||
|
||||
uint16_t date_get_age(struct date *birth_date, struct date *current_date) {
|
||||
uint16_t age = current_date->year - birth_date->year;
|
||||
if (current_date->month < birth_date->month || (current_date->month == birth_date->month && current_date->day < birth_date->day)) {
|
||||
age--;
|
||||
}
|
||||
return age;
|
||||
}
|
350
lib/date.h
350
lib/date.h
@ -1,350 +0,0 @@
|
||||
#ifndef __LIBCPROJECT_DATE__
|
||||
#define __LIBCPROJECT_DATE__
|
||||
|
||||
#include <errno.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdint.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <time.h>
|
||||
|
||||
#include "convert.h"
|
||||
#include "mathematics.h"
|
||||
#include "string.h"
|
||||
#include "types.h"
|
||||
|
||||
#define SECONDS_PER_MINUTE 60
|
||||
#define SECONDS_PER_HOUR (60 * SECONDS_PER_MINUTE)
|
||||
#define SECONDS_PER_DAY (24 * SECONDS_PER_HOUR)
|
||||
#define MILLISECONDS_PER_SECOND 1000
|
||||
|
||||
/**
|
||||
* @brief Date object representing a single moment in time.
|
||||
* @since v4.3.0
|
||||
*/
|
||||
struct date {
|
||||
/**
|
||||
* Year.
|
||||
* Between [0, 9999] (inclusive).
|
||||
* As per ISO 8601, a four-digit year [YYYY] and represents years from 0000 to 9999, year 0000 being equal to 1 BC and all others AD.
|
||||
*/
|
||||
uint16_t year;
|
||||
|
||||
/**
|
||||
* Month.
|
||||
* Between [1, 12] (inclusive).
|
||||
*/
|
||||
uint8_t month;
|
||||
|
||||
/**
|
||||
* Day.
|
||||
* Between [1, 31] (inclusive).
|
||||
*/
|
||||
uint8_t day;
|
||||
|
||||
/**
|
||||
* Hours.
|
||||
* Between [0, 23] (inclusive).
|
||||
*/
|
||||
uint8_t hours;
|
||||
|
||||
/**
|
||||
* Minutes.
|
||||
* Between [0, 59] (inclusive).
|
||||
*/
|
||||
uint8_t minutes;
|
||||
|
||||
/**
|
||||
* Seconds.
|
||||
* Between [0, 59] (inclusive).
|
||||
*/
|
||||
uint8_t seconds;
|
||||
|
||||
/**
|
||||
* Milliseconds.
|
||||
* Between [0, 999] (inclusive).
|
||||
*/
|
||||
uint16_t milliseconds;
|
||||
|
||||
/**
|
||||
* Timezone UTC offset.
|
||||
* Between [-12, 14]
|
||||
*/
|
||||
int8_t timezone_utc_offset;
|
||||
};
|
||||
|
||||
/**
|
||||
* @brief Return the copy of a date.
|
||||
*
|
||||
* @param date
|
||||
* @return struct date*
|
||||
* @since v4.3.0
|
||||
*/
|
||||
struct date *date_copy(struct date *date_to_copy);
|
||||
|
||||
/**
|
||||
* @brief Check if a year is valid, between [0, 9999] (inclusive).
|
||||
*
|
||||
* @param year
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid_year(uint16_t year);
|
||||
|
||||
/**
|
||||
* @brief Check if a month is valid, between [1, 12] (inclusive).
|
||||
*
|
||||
* @param month
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid_month(uint8_t month);
|
||||
|
||||
/**
|
||||
* @brief Check if a day is valid, between [1, 31] (inclusive).
|
||||
*
|
||||
* @param day
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid_day(uint8_t day);
|
||||
|
||||
/**
|
||||
* @brief Check if hours are valid, between [0, 23] (inclusive).
|
||||
*
|
||||
* @param hours
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid_hours(uint8_t hours);
|
||||
|
||||
/**
|
||||
* @brief Check if minutes are valid, between [0, 59] (inclusive).
|
||||
*
|
||||
* @param minutes
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid_minutes(uint8_t minutes);
|
||||
|
||||
/**
|
||||
* @brief Check if seconds are valid, between [0, 59] (inclusive).
|
||||
*
|
||||
* @param seconds
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid_seconds(uint8_t seconds);
|
||||
|
||||
/**
|
||||
* @brief Check if milliseconds are valid, between [0, 999] (inclusive).
|
||||
*
|
||||
* @param milliseconds
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid_milliseconds(uint16_t milliseconds);
|
||||
|
||||
/**
|
||||
* @brief Check if the timezone UTC offset is valid, between [-12, 14] (inclusive).
|
||||
*
|
||||
* @param timezone_utc_offset
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid_timezone_utc_offset(int8_t timezone_utc_offset);
|
||||
|
||||
/**
|
||||
* @brief Check if the date is valid (all fields are possible).
|
||||
*
|
||||
* @param date
|
||||
* @return bool
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_valid(struct date *date);
|
||||
|
||||
/**
|
||||
* @brief String representing the date in the date time string format, a simplified format based on ISO 8601, which is always 24 characters long (`YYYY-MM-DDTHH:mm:ss.sssZ`). The timezone is always UTC, as denoted by the suffix `Z`.
|
||||
*
|
||||
* @param date
|
||||
* @return string_t
|
||||
*
|
||||
* @code
|
||||
* date_to_iso_string() // "2024-09-11T09:39:18.203Z"
|
||||
* @endcode
|
||||
*
|
||||
* @since v4.3.0
|
||||
*/
|
||||
string_t date_to_iso_string(struct date *date);
|
||||
|
||||
/**
|
||||
* @brief String representing the date in the ISO 8601 format, without time information (`YYYY-MM-DD`).
|
||||
*
|
||||
* @param date
|
||||
* @return string_t
|
||||
*
|
||||
* @code
|
||||
* date_to_iso_string_without_time() // "2024-09-11"
|
||||
* @endcode
|
||||
*
|
||||
* @since v4.3.0
|
||||
*/
|
||||
string_t date_to_iso_string_without_time(struct date *date);
|
||||
|
||||
/**
|
||||
* @brief Create a date from an ISO 8601 string, with the format `YYYY-MM-DDTHH:mm:ss.sssZ`.
|
||||
*
|
||||
* The timezone is always UTC, as denoted by the suffix `Z`.
|
||||
*
|
||||
* @param iso_string
|
||||
* @return struct date*
|
||||
* @since v4.3.0
|
||||
*/
|
||||
struct date *date_from_iso_string(string_t iso_string);
|
||||
|
||||
/**
|
||||
* @brief Get number of days in one month [1, 12].
|
||||
*
|
||||
* @param month
|
||||
* @return uint8_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
uint8_t date_get_days_of_month(uint8_t month, uint16_t year);
|
||||
|
||||
/**
|
||||
* @brief Determine if a year is a leap year.
|
||||
*
|
||||
* @param year
|
||||
* @return bool
|
||||
*
|
||||
* @code
|
||||
* date_is_leap_year(2020) // true
|
||||
*
|
||||
* date_is_leap_year(2021) // false
|
||||
*
|
||||
* date_is_leap_year(2022) // false
|
||||
*
|
||||
* date_is_leap_year(2023) // false
|
||||
*
|
||||
* date_is_leap_year(2024) // true
|
||||
* @endcode
|
||||
*
|
||||
* @since v4.3.0
|
||||
*/
|
||||
bool date_get_is_leap_year(uint16_t year);
|
||||
|
||||
/**
|
||||
* @brief Convert milliseconds to seconds.
|
||||
*
|
||||
* @param milliseconds
|
||||
* @return uint64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
uint64_t date_convert_milliseconds_to_seconds(uint16_t milliseconds);
|
||||
|
||||
/**
|
||||
* @brief Convert seconds to milliseconds.
|
||||
*
|
||||
* @param seconds
|
||||
* @return uint64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
uint64_t date_convert_seconds_to_milliseconds(uint64_t seconds);
|
||||
|
||||
/**
|
||||
* @brief Convert days to seconds.
|
||||
*
|
||||
* @param days
|
||||
* @return uint64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
uint64_t date_convert_days_to_seconds(uint64_t days);
|
||||
|
||||
/**
|
||||
* @brief Convert hours, minutes, and seconds to seconds.
|
||||
*
|
||||
* @param hours
|
||||
* @param minutes
|
||||
* @param seconds
|
||||
* @return uint64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
uint64_t date_convert_hms_to_seconds(uint8_t hours, uint8_t minutes, uint8_t seconds);
|
||||
|
||||
/**
|
||||
* @brief Convert a date to total seconds.
|
||||
*
|
||||
* @param date
|
||||
* @return uint64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
uint64_t date_to_total_seconds(struct date *date);
|
||||
|
||||
/**
|
||||
* @brief Calculate the duration in seconds between 2 dates.
|
||||
*
|
||||
* @param date1
|
||||
* @param date2
|
||||
* @return uint64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
uint64_t date_duration_seconds_between_2_dates(struct date *date1, struct date *date2);
|
||||
|
||||
/**
|
||||
* @brief Add hours to the date, managing the day, month, year changes if necessary.
|
||||
*
|
||||
* NOTE: Mutates the date.
|
||||
*
|
||||
* @param date
|
||||
* @param hours
|
||||
* @since v4.3.0
|
||||
*/
|
||||
void date_add_hours(struct date *date, int64_t hours);
|
||||
|
||||
/**
|
||||
* @brief Adds days to the date, managing month and year changes as needed.
|
||||
*
|
||||
* NOTE: Mutates the date.
|
||||
*
|
||||
* @param date The date to which days are being added.
|
||||
* @param days The number of days to add.
|
||||
* @since v4.3.0
|
||||
*/
|
||||
void date_add_days(struct date *date, int64_t days);
|
||||
|
||||
/**
|
||||
* @brief Transform the date with a Timezone UTC Offset to UTC (timezone_utc_offset = 0).
|
||||
*
|
||||
* NOTE: Mutates the date.
|
||||
*
|
||||
* @param date
|
||||
* @since v4.3.0
|
||||
*/
|
||||
void date_to_utc(struct date *date);
|
||||
|
||||
/**
|
||||
* @brief Get the current date in UTC.
|
||||
*
|
||||
* @return struct date*
|
||||
* @since v5.1.0
|
||||
*/
|
||||
struct date *date_get_now_utc();
|
||||
|
||||
/**
|
||||
* @brief Get the current date in local time.
|
||||
*
|
||||
* @return struct date*
|
||||
* @since v5.1.0
|
||||
*/
|
||||
struct date *date_get_now_local();
|
||||
|
||||
/**
|
||||
* @brief Calculates the age of a person based on their birth date.
|
||||
*
|
||||
* @param birth_date
|
||||
* @return uint16_t
|
||||
* @since v5.1.0
|
||||
*/
|
||||
uint16_t date_get_age(struct date *birth_date, struct date *current_date);
|
||||
|
||||
#endif
|
@ -1,6 +1,6 @@
|
||||
#include "filesystem.h"
|
||||
|
||||
int filesystem_read(string_t path, byte_t **file_content, size_t *file_size) {
|
||||
int filesystem_read(string_t path, byte_t **file_content, off_t *file_size) {
|
||||
int file_descriptor = open(path, O_RDONLY);
|
||||
if (file_descriptor == -1) {
|
||||
return -1;
|
||||
@ -15,7 +15,7 @@ int filesystem_read(string_t path, byte_t **file_content, size_t *file_size) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
int filesystem_write(string_t path, byte_t *file_content, size_t file_size) {
|
||||
int filesystem_write(string_t path, byte_t *file_content, off_t file_size) {
|
||||
int file_descriptor = open(path, O_WRONLY | O_CREAT | O_TRUNC, 0644);
|
||||
if (file_descriptor == -1) {
|
||||
return -1;
|
||||
|
@ -24,7 +24,7 @@
|
||||
* @return int
|
||||
* @since v1.0.0
|
||||
*/
|
||||
int filesystem_read(string_t path, byte_t **file_content, size_t *file_size);
|
||||
int filesystem_read(string_t path, byte_t **file_content, off_t *file_size);
|
||||
|
||||
/**
|
||||
* @brief Write the content to a file.
|
||||
@ -37,13 +37,14 @@ int filesystem_read(string_t path, byte_t **file_content, size_t *file_size);
|
||||
* @return int
|
||||
* @since v1.0.0
|
||||
*/
|
||||
int filesystem_write(string_t path, byte_t *file_content, size_t file_size);
|
||||
int filesystem_write(string_t path, byte_t *file_content, off_t file_size);
|
||||
|
||||
/**
|
||||
* @brief Check if a path exists.
|
||||
*
|
||||
* @param path
|
||||
* @return bool
|
||||
* @return true
|
||||
* @return false
|
||||
* @since v3.1.0
|
||||
*/
|
||||
bool filesystem_exists(string_t path);
|
||||
|
@ -40,7 +40,7 @@ uint64_t hash(string_t key, size_t capacity) {
|
||||
memcpy(&m, message + offset, sizeof(uint64_t));
|
||||
v3 ^= m;
|
||||
|
||||
for (uint8_t i = 0; i < 2; i++) {
|
||||
for (int i = 0; i < 2; i++) {
|
||||
v0 = sip_round(v0, v1, v2, v3);
|
||||
v1 = ROTATE_LEFT(v1, 13);
|
||||
v2 = ROTATE_LEFT(v2, 16);
|
||||
@ -49,7 +49,7 @@ uint64_t hash(string_t key, size_t capacity) {
|
||||
}
|
||||
|
||||
v2 ^= 0xff;
|
||||
for (uint8_t i = 0; i < 4; i++) {
|
||||
for (int i = 0; i < 4; i++) {
|
||||
v0 = sip_round(v0, v1, v2, v3);
|
||||
v1 = ROTATE_LEFT(v1, 13);
|
||||
v2 = ROTATE_LEFT(v2, 16);
|
||||
@ -90,7 +90,7 @@ uint64_t hash(string_t key, size_t capacity) {
|
||||
|
||||
v3 ^= m;
|
||||
|
||||
for (uint8_t i = 0; i < 2; i++) {
|
||||
for (int i = 0; i < 2; i++) {
|
||||
v0 = sip_round(v0, v1, v2, v3);
|
||||
v1 = ROTATE_LEFT(v1, 13);
|
||||
v2 = ROTATE_LEFT(v2, 16);
|
||||
@ -100,7 +100,7 @@ uint64_t hash(string_t key, size_t capacity) {
|
||||
v0 ^= m;
|
||||
v2 ^= 0xff;
|
||||
|
||||
for (uint8_t i = 0; i < 4; i++) {
|
||||
for (int i = 0; i < 4; i++) {
|
||||
v0 = sip_round(v0, v1, v2, v3);
|
||||
v1 = ROTATE_LEFT(v1, 13);
|
||||
v2 = ROTATE_LEFT(v2, 16);
|
||||
|
@ -85,7 +85,8 @@ void *hash_map_get(struct hash_map *hash_map, string_t key);
|
||||
*
|
||||
* @param hash_map
|
||||
* @param key
|
||||
* @return bool
|
||||
* @return true
|
||||
* @return false
|
||||
* @since v2.0.0
|
||||
*/
|
||||
bool hash_map_contains_key(struct hash_map *hash_map, string_t key);
|
||||
|
@ -1,23 +1,23 @@
|
||||
#include "mathematics.h"
|
||||
|
||||
bool mathematics_equals(const float64_t number1, const float64_t number2) {
|
||||
return (number1 - number2) < MATHEMATICS_DOUBLE_PRECISION;
|
||||
bool mathematics_equals(const float number1, const float number2) {
|
||||
return (number1 - number2) < MATHEMATICS_FLOAT_PRECISION;
|
||||
}
|
||||
|
||||
uint64_t mathematics_absolute_value(const int64_t number) {
|
||||
unsigned long long mathematics_absolute_value(const long long number) {
|
||||
if (number >= 0) {
|
||||
return number;
|
||||
}
|
||||
return -number;
|
||||
}
|
||||
|
||||
uint64_t mathematics_pow(uint64_t base, uint64_t exponent) {
|
||||
unsigned long long mathematics_pow(unsigned long long base, unsigned long long exponent) {
|
||||
return exponent == 0 ? 1 : base * mathematics_pow(base, exponent - 1);
|
||||
}
|
||||
|
||||
float64_t mathematics_root(float64_t number, uint64_t nth_root) {
|
||||
float64_t result = number;
|
||||
float64_t previous_result = 0;
|
||||
float mathematics_root(float number, unsigned int nth_root) {
|
||||
float result = number;
|
||||
float previous_result = 0;
|
||||
while (!mathematics_equals(result, previous_result)) {
|
||||
result = (((nth_root - 1) * previous_result) + (number / mathematics_pow(result, nth_root - 1))) / nth_root;
|
||||
previous_result = result;
|
||||
@ -25,46 +25,10 @@ float64_t mathematics_root(float64_t number, uint64_t nth_root) {
|
||||
return result;
|
||||
}
|
||||
|
||||
float64_t mathematics_square_root(float64_t number) {
|
||||
float mathematics_square_root(float number) {
|
||||
return mathematics_root(number, 2);
|
||||
}
|
||||
|
||||
uint64_t mathematics_factorial(uint64_t number) {
|
||||
unsigned long long mathematics_factorial(unsigned long long number) {
|
||||
return number == 0 ? 1 : number * mathematics_factorial(number - 1);
|
||||
}
|
||||
|
||||
int64_t mathematics_opposite(int64_t number) {
|
||||
return number * -1;
|
||||
}
|
||||
|
||||
int64_t mathematics_max(int64_t number1, int64_t number2) {
|
||||
return number1 > number2 ? number1 : number2;
|
||||
}
|
||||
|
||||
int64_t mathematics_max_values(int64_t *values, size_t values_length) {
|
||||
int64_t max = 0;
|
||||
if (values_length <= 0) {
|
||||
return max;
|
||||
}
|
||||
max = values[0];
|
||||
for (size_t index = 1; index < values_length; index++) {
|
||||
max = mathematics_max(max, values[index]);
|
||||
}
|
||||
return max;
|
||||
}
|
||||
|
||||
int64_t mathematics_min(int64_t number1, int64_t number2) {
|
||||
return number1 > number2 ? number2 : number1;
|
||||
}
|
||||
|
||||
int64_t mathematics_min_values(int64_t *values, size_t values_length) {
|
||||
int64_t min = 0;
|
||||
if (values_length <= 0) {
|
||||
return min;
|
||||
}
|
||||
min = values[0];
|
||||
for (size_t index = 1; index < values_length; index++) {
|
||||
min = mathematics_min(min, values[index]);
|
||||
}
|
||||
return min;
|
||||
}
|
||||
|
@ -1,11 +1,10 @@
|
||||
#ifndef __LIBCPROJECT_MATHEMATICS__
|
||||
#define __LIBCPROJECT_MATHEMATICS__
|
||||
|
||||
#define MATHEMATICS_DOUBLE_PRECISION 0.0000000001
|
||||
#define MATHEMATICS_FLOAT_PRECISION 0.00000001
|
||||
|
||||
#include <errno.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "types.h"
|
||||
|
||||
@ -14,111 +13,57 @@
|
||||
*
|
||||
* @param number1
|
||||
* @param number2
|
||||
* @return bool
|
||||
* @return true
|
||||
* @return false
|
||||
* @since v1.0.0
|
||||
*/
|
||||
bool mathematics_equals(const float64_t number1, const float64_t number2);
|
||||
bool mathematics_equals(const float number1, const float number2);
|
||||
|
||||
/**
|
||||
* @brief Get the absolute value of a number.
|
||||
*
|
||||
* @param number
|
||||
* @return uint64_t
|
||||
* @return unsigned long long
|
||||
* @since v1.0.0
|
||||
*/
|
||||
uint64_t mathematics_absolute_value(const int64_t number);
|
||||
unsigned long long mathematics_absolute_value(const long long number);
|
||||
|
||||
/**
|
||||
* @brief Calculates the power of a number.
|
||||
*
|
||||
* @param base
|
||||
* @param exponent
|
||||
* @return uint64_t
|
||||
* @return unsigned long long
|
||||
* @since v1.0.0
|
||||
*/
|
||||
uint64_t mathematics_pow(uint64_t base, uint64_t exponent);
|
||||
unsigned long long mathematics_pow(unsigned long long base, unsigned long long exponent);
|
||||
|
||||
/**
|
||||
* @brief Calculates the nth root of a number.
|
||||
*
|
||||
* @param number
|
||||
* @param nth_root
|
||||
* @return float64_t
|
||||
* @return float
|
||||
* @since v1.0.0
|
||||
*/
|
||||
float64_t mathematics_root(float64_t number, uint64_t nth_root);
|
||||
float mathematics_root(float number, unsigned int nth_root);
|
||||
|
||||
/**
|
||||
* @brief Calculates the square root of a number using Heron's method.
|
||||
*
|
||||
* @param number
|
||||
* @return float64_t
|
||||
* @return float
|
||||
* @since v1.0.0
|
||||
*/
|
||||
float64_t mathematics_square_root(float64_t number);
|
||||
float mathematics_square_root(float number);
|
||||
|
||||
/**
|
||||
* @brief Calculates the factorial of a number.
|
||||
*
|
||||
* @param number
|
||||
* @return uint64_t
|
||||
* @return unsigned long long
|
||||
* @since v1.0.0
|
||||
*/
|
||||
uint64_t mathematics_factorial(uint64_t number);
|
||||
|
||||
/**
|
||||
* @brief Calulcates the opposite number (additive inverse).
|
||||
*
|
||||
* @param number
|
||||
* @return int64_t
|
||||
*
|
||||
* @code
|
||||
* mathematics_opposite(7) // -7
|
||||
*
|
||||
* mathematics_opposite(-7) // 7
|
||||
* @endcode
|
||||
* @since v4.3.0
|
||||
*/
|
||||
int64_t mathematics_opposite(int64_t number);
|
||||
|
||||
/**
|
||||
* @brief Returns the largest number between 2 numbers.
|
||||
*
|
||||
* @param number1
|
||||
* @param number2
|
||||
* @return int64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
int64_t mathematics_max(int64_t number1, int64_t number2);
|
||||
|
||||
/**
|
||||
* @brief Returns the largest number between multiple numbers. If the array is empty, returns 0.
|
||||
*
|
||||
* @param values
|
||||
* @param values_length
|
||||
* @return int64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
int64_t mathematics_max_values(int64_t *values, size_t values_length);
|
||||
|
||||
/**
|
||||
* @brief Returns the smallest number between 2 numbers.
|
||||
*
|
||||
* @param number1
|
||||
* @param number2
|
||||
* @return int64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
int64_t mathematics_min(int64_t number1, int64_t number2);
|
||||
|
||||
/**
|
||||
* @brief Returns the smallest number between multiple numbers. If the array is empty, returns 0.
|
||||
*
|
||||
* @param values
|
||||
* @param values_length
|
||||
* @return int64_t
|
||||
* @since v4.3.0
|
||||
*/
|
||||
int64_t mathematics_min_values(int64_t *values, size_t values_length);
|
||||
unsigned long long mathematics_factorial(unsigned long long number);
|
||||
|
||||
#endif
|
||||
|
35
lib/string.c
35
lib/string.c
@ -296,7 +296,7 @@ bool string_get_is_substring(const string_t string, const string_t substring) {
|
||||
return is_substring;
|
||||
}
|
||||
|
||||
string_t string_get_formatted_number(const int64_t number, string_t separator) {
|
||||
string_t string_get_formatted_number(const long long number, string_t separator) {
|
||||
string_t number_string_temp = convert_number_to_string(number);
|
||||
string_t number_string = number_string_temp;
|
||||
bool is_negative = number_string_temp[0] == '-';
|
||||
@ -407,36 +407,3 @@ size_t string_last_position_of(const string_t string, const char character) {
|
||||
}
|
||||
return position_found;
|
||||
}
|
||||
|
||||
string_t string_pad_start(const string_t string, const string_t pad_string, size_t target_length) {
|
||||
string_t result = malloc(sizeof(char) * (target_length + 1));
|
||||
size_t initial_length = string_get_length(string);
|
||||
size_t left_length = target_length - initial_length;
|
||||
if (target_length <= initial_length) {
|
||||
left_length = 0;
|
||||
}
|
||||
size_t pad_length = string_get_length(pad_string);
|
||||
size_t count_pad_string = 0;
|
||||
size_t index_initial_string = 0;
|
||||
for (size_t index = 0; index < target_length; index++) {
|
||||
if (index < left_length) {
|
||||
size_t index_pad_string = count_pad_string % pad_length;
|
||||
result[index] = pad_string[index_pad_string];
|
||||
count_pad_string += 1;
|
||||
} else {
|
||||
result[index] = string[index_initial_string];
|
||||
index_initial_string += 1;
|
||||
}
|
||||
}
|
||||
result[target_length] = '\0';
|
||||
return result;
|
||||
}
|
||||
|
||||
string_t string_zero_pad(uint64_t number, size_t places) {
|
||||
string_t number_string = convert_number_to_string((int64_t)number);
|
||||
string_t pad_string = string_copy("0");
|
||||
string_t result = string_pad_start(number_string, pad_string, places);
|
||||
free(pad_string);
|
||||
free(number_string);
|
||||
return result;
|
||||
}
|
||||
|
45
lib/string.h
45
lib/string.h
@ -224,15 +224,13 @@ bool string_get_is_substring(const string_t string, const string_t substring);
|
||||
* @param number
|
||||
* @param separator
|
||||
* @return string_t
|
||||
*
|
||||
* @code
|
||||
* string_get_formatted_number(1000, " ") // "1 000"
|
||||
*
|
||||
* string_get_formatted_number(1000, ",") // "1,000"
|
||||
* @endcode
|
||||
* @since v1.0.0
|
||||
*/
|
||||
string_t string_get_formatted_number(const int64_t number, string_t separator);
|
||||
string_t string_get_formatted_number(const long long number, string_t separator);
|
||||
|
||||
/**
|
||||
* @brief Returns a pointer to the last occurrence of character in the string.
|
||||
@ -272,10 +270,7 @@ bool string_ends_with(const string_t string, const string_t prefix);
|
||||
* @param string
|
||||
* @param substring
|
||||
* @return size_t
|
||||
*
|
||||
* @code
|
||||
* string_position_of("hello world", 'e') // 2
|
||||
* @endcode
|
||||
* @example string_position_of("hello world", 'e') // 2
|
||||
* @since v4.2.0
|
||||
*/
|
||||
size_t string_position_of(const string_t string, const char character);
|
||||
@ -286,43 +281,9 @@ size_t string_position_of(const string_t string, const char character);
|
||||
* @param string
|
||||
* @param character
|
||||
* @return size_t
|
||||
*
|
||||
* @code
|
||||
* string_last_position_of("hello world", 'o') // 8
|
||||
* @endcode
|
||||
* @example string_last_position_of("hello world", 'o') // 8
|
||||
* @since v4.2.0
|
||||
*/
|
||||
size_t string_last_position_of(const string_t string, const char character);
|
||||
|
||||
/**
|
||||
* @brief Pads a `string` with another `pad_string` (multiple times, if needed) until the resulting string reaches the `target_length`. The padding is applied from the start (left) of the string.
|
||||
*
|
||||
* @param string The string to pad.
|
||||
* @param pad_string The string to pad the current string with, to the left.
|
||||
* @param target_length
|
||||
* @return string_t
|
||||
*
|
||||
* @code
|
||||
* string_pad_start("hello", " ", 10) // " hello"
|
||||
* @endcode
|
||||
* @since v4.3.0
|
||||
*/
|
||||
string_t string_pad_start(const string_t string, const string_t pad_string, size_t target_length);
|
||||
|
||||
/**
|
||||
* @brief Pad a number with zeros.
|
||||
*
|
||||
* @param number
|
||||
* @param places
|
||||
* @return string_t
|
||||
*
|
||||
* @code
|
||||
* string_zero_pad(1, 2) // "01"
|
||||
*
|
||||
* string_zero_pad(10, 2) // "10"
|
||||
* @endcode
|
||||
* @since v4.3.0
|
||||
*/
|
||||
string_t string_zero_pad(uint64_t number, size_t places);
|
||||
|
||||
#endif
|
||||
|
@ -21,66 +21,6 @@ string_t terminal_input() {
|
||||
return string;
|
||||
}
|
||||
|
||||
void terminal_print_int(void* value) {
|
||||
printf("%d", *(int*)value);
|
||||
}
|
||||
|
||||
void terminal_print_bool(void* value) {
|
||||
printf("%s", *(bool*)value ? "true" : "false");
|
||||
}
|
||||
|
||||
void terminal_print_long(void* value) {
|
||||
printf("%ld", *(long*)value);
|
||||
}
|
||||
|
||||
void terminal_print_unsigned_long(void* value) {
|
||||
printf("%lu", *(unsigned long*)value);
|
||||
}
|
||||
|
||||
void terminal_print_float(void* value) {
|
||||
printf("%f", *(float*)value);
|
||||
}
|
||||
|
||||
void terminal_print_double(void* value) {
|
||||
printf("%f", *(double*)value);
|
||||
}
|
||||
|
||||
void terminal_print_int8_t(void* value) {
|
||||
printf("%d", *(int8_t*)value);
|
||||
}
|
||||
|
||||
void terminal_print_int16_t(void* value) {
|
||||
printf("%d", *(int16_t*)value);
|
||||
}
|
||||
|
||||
void terminal_print_int32_t(void* value) {
|
||||
printf("%d", *(int32_t*)value);
|
||||
}
|
||||
|
||||
void terminal_print_int64_t(void* value) {
|
||||
printf("%ld", *(int64_t*)value);
|
||||
}
|
||||
|
||||
void terminal_print_uint8_t(void* value) {
|
||||
printf("%u", *(uint8_t*)value);
|
||||
}
|
||||
|
||||
void terminal_print_uint16_t(void* value) {
|
||||
printf("%u", *(uint16_t*)value);
|
||||
}
|
||||
|
||||
void terminal_print_uint32_t(void* value) {
|
||||
printf("%u", *(uint32_t*)value);
|
||||
}
|
||||
|
||||
void terminal_print_uint64_t(void* value) {
|
||||
printf("%lu", *(uint64_t*)value);
|
||||
}
|
||||
|
||||
void terminal_print_char(void* value) {
|
||||
printf("%c", *(string_t)value);
|
||||
}
|
||||
|
||||
void terminal_print_array(void* array, size_t array_size, size_t element_size, void (*print_element)(void*)) {
|
||||
printf("[");
|
||||
for (size_t index = 0; index < array_size; index++) {
|
||||
@ -94,6 +34,22 @@ void terminal_print_array(void* array, size_t array_size, size_t element_size, v
|
||||
printf("]\n");
|
||||
}
|
||||
|
||||
void terminal_print_int(void* value) {
|
||||
printf("%d", *(int*)value);
|
||||
}
|
||||
|
||||
void terminal_print_long(void* value) {
|
||||
printf("%ld", *(long*)value);
|
||||
}
|
||||
|
||||
void terminal_print_unsigned_long(void* value) {
|
||||
printf("%lu", *(unsigned long*)value);
|
||||
}
|
||||
|
||||
void terminal_print_char(void* value) {
|
||||
printf("%c", *(string_t)value);
|
||||
}
|
||||
|
||||
void terminal_print_string(void* value) {
|
||||
printf("%s", (string_t)value);
|
||||
}
|
||||
@ -108,7 +64,7 @@ void terminal_print_stack(struct stack* stack, void (*print_element)(void*)) {
|
||||
while (node_current != NULL) {
|
||||
printf("|\t");
|
||||
void* element = node_current->data;
|
||||
print_element(element);
|
||||
print_element(&element);
|
||||
node_current = node_current->next;
|
||||
printf("\t|\n");
|
||||
}
|
||||
@ -124,7 +80,7 @@ void terminal_print_queue(struct queue* queue, void (*print_element)(void*)) {
|
||||
while (node_current != NULL) {
|
||||
printf("|\t");
|
||||
void* element = node_current->data;
|
||||
print_element(element);
|
||||
print_element(&element);
|
||||
node_current = node_current->next;
|
||||
printf("\t|\n");
|
||||
}
|
||||
@ -140,7 +96,7 @@ void terminal_print_linked_list(struct linked_list* linked_list, void (*print_el
|
||||
while (node_current != NULL) {
|
||||
void* element = (string_t)node_current->data;
|
||||
node_current = node_current->next;
|
||||
print_element(element);
|
||||
print_element(&element);
|
||||
printf(" -> ");
|
||||
}
|
||||
printf("NULL\n");
|
||||
@ -160,21 +116,12 @@ void terminal_print_hash_map(struct hash_map* hash_map, void (*print_element)(vo
|
||||
printf("\t\"");
|
||||
terminal_print_string(key);
|
||||
printf("\" -> ");
|
||||
print_element(value);
|
||||
print_element(&value);
|
||||
printf("\n");
|
||||
}
|
||||
printf("}\n");
|
||||
free(keys);
|
||||
}
|
||||
|
||||
void terminal_print_array_list(struct array_list* list, void (*print_element)(void*)) {
|
||||
printf("[");
|
||||
for (size_t index = 0; index < list->size; index++) {
|
||||
void* element = list->data[index];
|
||||
print_element(element);
|
||||
if (index < list->size - 1) {
|
||||
printf(", ");
|
||||
}
|
||||
}
|
||||
printf("]\n");
|
||||
terminal_print_array(list->data, list->size, sizeof(void*), print_element);
|
||||
}
|
||||
|
110
lib/terminal.h
110
lib/terminal.h
@ -23,6 +23,17 @@
|
||||
*/
|
||||
string_t terminal_input();
|
||||
|
||||
/**
|
||||
* @brief Print an array.
|
||||
*
|
||||
* @param array
|
||||
* @param array_size
|
||||
* @param element_size
|
||||
* @param print_element
|
||||
* @since v1.0.0
|
||||
*/
|
||||
void terminal_print_array(void* array, size_t array_size, size_t element_size, void (*print_element)(void*));
|
||||
|
||||
/**
|
||||
* @brief Print a int.
|
||||
*
|
||||
@ -31,14 +42,6 @@ string_t terminal_input();
|
||||
*/
|
||||
void terminal_print_int(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a boolean.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_bool(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a long.
|
||||
*
|
||||
@ -55,86 +58,6 @@ void terminal_print_long(void* value);
|
||||
*/
|
||||
void terminal_print_unsigned_long(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a float.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_float(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a double.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_double(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a int8_t.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_int8_t(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a int16_t.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_int16_t(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a int32_t.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_int32_t(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a int64_t.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_int64_t(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a uint8_t.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_uint8_t(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a uint16_t.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_uint16_t(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a uint32_t.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_uint32_t(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a uint64_t.
|
||||
*
|
||||
* @param value
|
||||
* @since v5.0.0
|
||||
*/
|
||||
void terminal_print_uint64_t(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print a char.
|
||||
*
|
||||
@ -143,17 +66,6 @@ void terminal_print_uint64_t(void* value);
|
||||
*/
|
||||
void terminal_print_char(void* value);
|
||||
|
||||
/**
|
||||
* @brief Print an array.
|
||||
*
|
||||
* @param array
|
||||
* @param array_size
|
||||
* @param element_size
|
||||
* @param print_element
|
||||
* @since v1.0.0
|
||||
*/
|
||||
void terminal_print_array(void* array, size_t array_size, size_t element_size, void (*print_element)(void*));
|
||||
|
||||
/**
|
||||
* @brief Print a string.
|
||||
*
|
||||
|
@ -7,7 +7,4 @@ typedef uint8_t byte_t;
|
||||
|
||||
typedef char* string_t;
|
||||
|
||||
typedef float float32_t;
|
||||
typedef double float64_t;
|
||||
|
||||
#endif
|
||||
|
@ -2,10 +2,8 @@
|
||||
#define __LIBCPROJECT__
|
||||
|
||||
#include "lib/array_list.h"
|
||||
#include "lib/assert.h"
|
||||
#include "lib/character.h"
|
||||
#include "lib/convert.h"
|
||||
#include "lib/date.h"
|
||||
#include "lib/filesystem.h"
|
||||
#include "lib/hash_map.h"
|
||||
#include "lib/linked_list.h"
|
||||
|
131
main.c
131
main.c
@ -4,135 +4,8 @@
|
||||
#include "libcproject.h"
|
||||
|
||||
int main() {
|
||||
int integer = 5;
|
||||
terminal_print_int(&integer);
|
||||
printf("\n");
|
||||
|
||||
bool boolean = true;
|
||||
printf("terminal_print_bool: ");
|
||||
terminal_print_bool(&boolean);
|
||||
printf(", ");
|
||||
boolean = false;
|
||||
terminal_print_bool(&boolean);
|
||||
printf("\n");
|
||||
|
||||
long integer_long = 5;
|
||||
printf("terminal_print_long: ");
|
||||
terminal_print_long(&integer_long);
|
||||
printf("\n");
|
||||
|
||||
unsigned long integer_unsigned_long = 5;
|
||||
printf("terminal_print_unsigned_long: ");
|
||||
terminal_print_unsigned_long(&integer_unsigned_long);
|
||||
printf("\n");
|
||||
|
||||
float floating_point = 5.5;
|
||||
printf("terminal_print_float: ");
|
||||
terminal_print_float(&floating_point);
|
||||
printf("\n");
|
||||
|
||||
double floating_point_double = 5.5;
|
||||
printf("terminal_print_double: ");
|
||||
terminal_print_double(&floating_point_double);
|
||||
printf("\n");
|
||||
|
||||
int8_t integer_8 = 5;
|
||||
printf("terminal_print_int8_t: ");
|
||||
terminal_print_int8_t(&integer_8);
|
||||
printf("\n");
|
||||
|
||||
int16_t integer_16 = 5;
|
||||
printf("terminal_print_int16_t: ");
|
||||
terminal_print_int16_t(&integer_16);
|
||||
printf("\n");
|
||||
|
||||
int32_t integer_32 = 5;
|
||||
printf("terminal_print_int32_t: ");
|
||||
terminal_print_int32_t(&integer_32);
|
||||
printf("\n");
|
||||
|
||||
int64_t integer_64 = 5;
|
||||
printf("terminal_print_int64_t: ");
|
||||
terminal_print_int64_t(&integer_64);
|
||||
printf("\n");
|
||||
|
||||
uint8_t integer_unsigned_8 = 5;
|
||||
printf("terminal_print_uint8_t: ");
|
||||
terminal_print_uint8_t(&integer_unsigned_8);
|
||||
printf("\n");
|
||||
|
||||
uint16_t integer_unsigned_16 = 5;
|
||||
printf("terminal_print_uint16_t: ");
|
||||
terminal_print_uint16_t(&integer_unsigned_16);
|
||||
printf("\n");
|
||||
|
||||
uint32_t integer_unsigned_32 = 5;
|
||||
printf("terminal_print_uint32_t: ");
|
||||
terminal_print_uint32_t(&integer_unsigned_32);
|
||||
printf("\n");
|
||||
|
||||
uint64_t integer_unsigned_64 = 5;
|
||||
printf("terminal_print_uint64_t: ");
|
||||
terminal_print_uint64_t(&integer_unsigned_64);
|
||||
printf("\n");
|
||||
|
||||
char character = 'c';
|
||||
printf("terminal_print_char: ");
|
||||
terminal_print_char(&character);
|
||||
printf("\n");
|
||||
|
||||
int array[] = {1, 2, 3, 4, 5};
|
||||
printf("terminal_print_array: ");
|
||||
terminal_print_array(array, 5, sizeof(int), terminal_print_int);
|
||||
|
||||
string_t string = "Hello, world!";
|
||||
printf("terminal_print_string: ");
|
||||
terminal_print_string(string);
|
||||
printf("\n");
|
||||
|
||||
struct stack* stack = stack_initialization();
|
||||
stack_push(stack, &integer);
|
||||
printf("terminal_print_stack: ");
|
||||
terminal_print_stack(stack, terminal_print_int);
|
||||
stack_free(stack);
|
||||
|
||||
struct queue* queue = queue_initialization();
|
||||
queue_push(queue, &integer);
|
||||
printf("terminal_print_queue: ");
|
||||
terminal_print_queue(queue, terminal_print_int);
|
||||
queue_free(queue);
|
||||
|
||||
struct linked_list* linked_list = linked_list_initialization();
|
||||
linked_list_add_after_last(linked_list, string);
|
||||
linked_list_add_after_last(linked_list, string);
|
||||
printf("terminal_print_linked_list: ");
|
||||
terminal_print_linked_list(linked_list, terminal_print_string);
|
||||
linked_list_free(linked_list);
|
||||
|
||||
struct hash_map* hash_map = hash_map_initialization();
|
||||
hash_map_add(hash_map, "key", &integer);
|
||||
printf("terminal_print_hash_map: ");
|
||||
terminal_print_hash_map(hash_map, terminal_print_int);
|
||||
hash_map_free(hash_map);
|
||||
|
||||
struct array_list* array_list = array_list_initialization();
|
||||
array_list_add(array_list, &integer);
|
||||
array_list_add(array_list, &integer);
|
||||
printf("terminal_print_array_list: ");
|
||||
terminal_print_array_list(array_list, terminal_print_int);
|
||||
array_list_free(array_list);
|
||||
|
||||
struct date* date = date_get_now_utc();
|
||||
string_t iso_string = date_to_iso_string(date);
|
||||
printf("date_get_now_utc = %s\n", iso_string);
|
||||
free(iso_string);
|
||||
free(date);
|
||||
|
||||
date = date_get_now_local();
|
||||
iso_string = date_to_iso_string(date);
|
||||
printf("date_get_now_local = %s\n", iso_string);
|
||||
free(iso_string);
|
||||
free(date);
|
||||
|
||||
printf("%s\n", string);
|
||||
printf("string_length = %ld\n", string_get_length(string));
|
||||
return EXIT_SUCCESS;
|
||||
}
|
||||
|
@ -6,6 +6,7 @@
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "libcproject.h"
|
||||
#include "test.h"
|
||||
|
||||
void character_test();
|
||||
|
||||
|
@ -6,6 +6,7 @@
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "libcproject.h"
|
||||
#include "test.h"
|
||||
|
||||
void convert_test();
|
||||
|
||||
|
191
test/date_test.c
191
test/date_test.c
@ -1,191 +0,0 @@
|
||||
#include "date_test.h"
|
||||
|
||||
void date_test() {
|
||||
date_copy_test();
|
||||
date_to_iso_string_test();
|
||||
date_to_iso_string_without_time_test();
|
||||
date_from_iso_string_test();
|
||||
date_get_is_leap_year_test();
|
||||
date_duration_seconds_between_2_dates_test();
|
||||
date_to_utc_test();
|
||||
date_get_age_test();
|
||||
}
|
||||
|
||||
void date_copy_test() {
|
||||
struct date *date = malloc(sizeof(struct date));
|
||||
date->year = 2024;
|
||||
date->month = 9;
|
||||
date->day = 10;
|
||||
date->hours = 20;
|
||||
date->minutes = 34;
|
||||
date->seconds = 25;
|
||||
date->milliseconds = 76;
|
||||
date->timezone_utc_offset = 0;
|
||||
|
||||
struct date *date2 = date_copy(date);
|
||||
assert(date != date2);
|
||||
assert(date->year == date2->year);
|
||||
assert(date->month == date2->month);
|
||||
assert(date->day == date2->day);
|
||||
assert(date->hours == date2->hours);
|
||||
assert(date->minutes == date2->minutes);
|
||||
assert(date->seconds == date2->seconds);
|
||||
assert(date->milliseconds == date2->milliseconds);
|
||||
assert(date->timezone_utc_offset == date2->timezone_utc_offset);
|
||||
|
||||
date->year = 2025;
|
||||
assert(date->year == 2025);
|
||||
assert(date2->year == 2024);
|
||||
|
||||
free(date);
|
||||
free(date2);
|
||||
}
|
||||
|
||||
void date_to_iso_string_test() {
|
||||
struct date *date = malloc(sizeof(struct date));
|
||||
date->year = 2024;
|
||||
date->month = 9;
|
||||
date->day = 10;
|
||||
date->hours = 20;
|
||||
date->minutes = 34;
|
||||
date->seconds = 25;
|
||||
date->milliseconds = 76;
|
||||
date->timezone_utc_offset = 0;
|
||||
|
||||
string_t iso_string = date_to_iso_string(date);
|
||||
assert(assert_string_equal(iso_string, "2024-09-10T20:34:25.076Z"));
|
||||
free(iso_string);
|
||||
|
||||
free(date);
|
||||
}
|
||||
|
||||
void date_to_iso_string_without_time_test() {
|
||||
struct date *date = malloc(sizeof(struct date));
|
||||
date->year = 2024;
|
||||
date->month = 9;
|
||||
date->day = 10;
|
||||
date->hours = 20;
|
||||
date->minutes = 34;
|
||||
date->seconds = 25;
|
||||
date->milliseconds = 76;
|
||||
date->timezone_utc_offset = 0;
|
||||
|
||||
string_t iso_string = date_to_iso_string_without_time(date);
|
||||
assert(assert_string_equal(iso_string, "2024-09-10"));
|
||||
free(iso_string);
|
||||
|
||||
free(date);
|
||||
}
|
||||
|
||||
void date_from_iso_string_test() {
|
||||
string_t iso_string = "2024-09-10T20:34:25.076Z";
|
||||
struct date *date = date_from_iso_string(iso_string);
|
||||
assert(date->year == 2024);
|
||||
assert(date->month == 9);
|
||||
assert(date->day == 10);
|
||||
assert(date->hours == 20);
|
||||
assert(date->minutes == 34);
|
||||
assert(date->seconds == 25);
|
||||
assert(date->milliseconds == 76);
|
||||
assert(date->timezone_utc_offset == 0);
|
||||
|
||||
free(date);
|
||||
}
|
||||
|
||||
void date_get_is_leap_year_test() {
|
||||
assert(date_get_is_leap_year(2020));
|
||||
assert(!date_get_is_leap_year(2021));
|
||||
assert(!date_get_is_leap_year(2022));
|
||||
assert(!date_get_is_leap_year(2023));
|
||||
assert(date_get_is_leap_year(2024));
|
||||
}
|
||||
|
||||
void date_duration_seconds_between_2_dates_test() {
|
||||
struct date *date1 = date_from_iso_string("2024-09-10T20:34:25.076Z");
|
||||
struct date *date2 = date_from_iso_string("2024-09-10T20:34:25.076Z");
|
||||
assert(date_duration_seconds_between_2_dates(date1, date2) == 0);
|
||||
free(date1);
|
||||
free(date2);
|
||||
|
||||
date1 = date_from_iso_string("2024-09-10T20:34:25.076Z");
|
||||
date2 = date_from_iso_string("2024-09-10T23:34:26.076Z");
|
||||
assert(date_duration_seconds_between_2_dates(date1, date2) == 10801);
|
||||
free(date1);
|
||||
free(date2);
|
||||
|
||||
date1 = date_from_iso_string("2024-09-10T20:34:25.076Z");
|
||||
date2 = date_from_iso_string("2024-09-10T20:48:25.076Z");
|
||||
assert(date_duration_seconds_between_2_dates(date1, date2) == 840);
|
||||
free(date1);
|
||||
free(date2);
|
||||
|
||||
date1 = date_from_iso_string("2024-09-10T20:34:25.076Z");
|
||||
date2 = date_from_iso_string("2024-09-10T20:34:38.076Z");
|
||||
assert(date_duration_seconds_between_2_dates(date1, date2) == 13);
|
||||
free(date1);
|
||||
free(date2);
|
||||
}
|
||||
|
||||
void date_to_utc_test() {
|
||||
struct date *date = date_from_iso_string("2024-09-10T20:34:25.076Z");
|
||||
date->timezone_utc_offset = 3;
|
||||
date_to_utc(date);
|
||||
assert(date->timezone_utc_offset == 0);
|
||||
string_t iso_string = date_to_iso_string(date);
|
||||
assert(assert_string_equal(iso_string, "2024-09-10T17:34:25.076Z"));
|
||||
free(iso_string);
|
||||
free(date);
|
||||
|
||||
date = date_from_iso_string("2024-09-10T20:34:25.076Z");
|
||||
date->timezone_utc_offset = -3;
|
||||
date_to_utc(date);
|
||||
assert(date->timezone_utc_offset == 0);
|
||||
iso_string = date_to_iso_string(date);
|
||||
assert(assert_string_equal(iso_string, "2024-09-10T23:34:25.076Z"));
|
||||
free(iso_string);
|
||||
free(date);
|
||||
|
||||
date = date_from_iso_string("2024-01-01T00:00:00.000Z");
|
||||
date->timezone_utc_offset = 3;
|
||||
date_to_utc(date);
|
||||
assert(date->timezone_utc_offset == 0);
|
||||
iso_string = date_to_iso_string(date);
|
||||
assert(assert_string_equal(iso_string, "2023-12-31T21:00:00.000Z"));
|
||||
free(iso_string);
|
||||
free(date);
|
||||
|
||||
date = date_from_iso_string("2023-12-31T21:00:00.000Z");
|
||||
date->timezone_utc_offset = -4;
|
||||
date_to_utc(date);
|
||||
assert(date->timezone_utc_offset == 0);
|
||||
iso_string = date_to_iso_string(date);
|
||||
assert(assert_string_equal(iso_string, "2024-01-01T01:00:00.000Z"));
|
||||
free(iso_string);
|
||||
free(date);
|
||||
}
|
||||
|
||||
void date_get_age_test() {
|
||||
struct date *birth_date = date_from_iso_string("1980-02-20T00:00:00.000Z");
|
||||
struct date *current_date = date_from_iso_string("2018-03-20T00:00:00.000Z");
|
||||
assert(date_get_age(birth_date, current_date) == 38);
|
||||
free(birth_date);
|
||||
free(current_date);
|
||||
|
||||
birth_date = date_from_iso_string("1980-07-20T00:00:00.000Z");
|
||||
current_date = date_from_iso_string("2018-03-20T00:00:00.000Z");
|
||||
assert(date_get_age(birth_date, current_date) == 37);
|
||||
free(birth_date);
|
||||
free(current_date);
|
||||
|
||||
birth_date = date_from_iso_string("1980-03-20T00:00:00.000Z");
|
||||
current_date = date_from_iso_string("2018-03-20T00:00:00.000Z");
|
||||
assert(date_get_age(birth_date, current_date) == 38);
|
||||
free(birth_date);
|
||||
free(current_date);
|
||||
|
||||
birth_date = date_from_iso_string("1980-03-25T00:00:00.000Z");
|
||||
current_date = date_from_iso_string("2018-03-20T00:00:00.000Z");
|
||||
assert(date_get_age(birth_date, current_date) == 37);
|
||||
free(birth_date);
|
||||
free(current_date);
|
||||
}
|
@ -1,26 +0,0 @@
|
||||
#ifndef __DATE_TEST__
|
||||
#define __DATE_TEST__
|
||||
|
||||
#include <assert.h>
|
||||
|
||||
#include "libcproject.h"
|
||||
|
||||
void date_test();
|
||||
|
||||
void date_copy_test();
|
||||
|
||||
void date_to_iso_string_test();
|
||||
|
||||
void date_to_iso_string_without_time_test();
|
||||
|
||||
void date_from_iso_string_test();
|
||||
|
||||
void date_get_is_leap_year_test();
|
||||
|
||||
void date_duration_seconds_between_2_dates_test();
|
||||
|
||||
void date_to_utc_test();
|
||||
|
||||
void date_get_age_test();
|
||||
|
||||
#endif
|
@ -4,7 +4,6 @@
|
||||
#include "array_list_test.h"
|
||||
#include "character_test.h"
|
||||
#include "convert_test.h"
|
||||
#include "date_test.h"
|
||||
#include "hash_map_test.h"
|
||||
#include "linked_list_test.h"
|
||||
#include "mathematics_test.h"
|
||||
@ -16,7 +15,6 @@ int main() {
|
||||
array_list_test();
|
||||
character_test();
|
||||
convert_test();
|
||||
date_test();
|
||||
hash_map_test();
|
||||
linked_list_test();
|
||||
mathematics_test();
|
||||
|
@ -6,11 +6,6 @@ void mathematics_test() {
|
||||
mathematics_root_test();
|
||||
mathematics_square_root_test();
|
||||
mathematics_factorial_test();
|
||||
mathematics_opposite_test();
|
||||
mathematics_max_test();
|
||||
mathematics_max_values_test();
|
||||
mathematics_min_test();
|
||||
mathematics_min_values_test();
|
||||
}
|
||||
|
||||
void mathematics_absolute_value_test() {
|
||||
@ -61,38 +56,3 @@ void mathematics_factorial_test() {
|
||||
assert(mathematics_factorial(9) == 362880);
|
||||
assert(mathematics_factorial(10) == 3628800);
|
||||
}
|
||||
|
||||
void mathematics_opposite_test() {
|
||||
assert(mathematics_opposite(-7) == 7);
|
||||
assert(mathematics_opposite(7) == -7);
|
||||
}
|
||||
|
||||
void mathematics_max_test() {
|
||||
assert(mathematics_max(0, 0) == 0);
|
||||
assert(mathematics_max(0, 1) == 1);
|
||||
assert(mathematics_max(2, 0) == 2);
|
||||
assert(mathematics_max(54, 37) == 54);
|
||||
}
|
||||
|
||||
void mathematics_max_values_test() {
|
||||
int64_t values[] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
|
||||
assert(mathematics_max_values(values, 10) == 9);
|
||||
|
||||
int64_t values2[] = {8, 6, 4, 7};
|
||||
assert(mathematics_max_values(values2, 4) == 8);
|
||||
}
|
||||
|
||||
void mathematics_min_test() {
|
||||
assert(mathematics_min(0, 0) == 0);
|
||||
assert(mathematics_min(3, 5) == 3);
|
||||
assert(mathematics_min(2, 1) == 1);
|
||||
assert(mathematics_min(54, 37) == 37);
|
||||
}
|
||||
|
||||
void mathematics_min_values_test() {
|
||||
int64_t values[] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
|
||||
assert(mathematics_min_values(values, 10) == 0);
|
||||
|
||||
int64_t values2[] = {9, 6, 8, 7};
|
||||
assert(mathematics_min_values(values2, 4) == 6);
|
||||
}
|
||||
|
@ -17,14 +17,4 @@ void mathematics_square_root_test();
|
||||
|
||||
void mathematics_factorial_test();
|
||||
|
||||
void mathematics_opposite_test();
|
||||
|
||||
void mathematics_max_test();
|
||||
|
||||
void mathematics_max_values_test();
|
||||
|
||||
void mathematics_min_test();
|
||||
|
||||
void mathematics_min_values_test();
|
||||
|
||||
#endif
|
||||
|
@ -27,8 +27,6 @@ void string_test() {
|
||||
string_ends_with_test();
|
||||
string_position_of_test();
|
||||
string_last_position_of_test();
|
||||
string_pad_start_test();
|
||||
string_zero_pad_test();
|
||||
}
|
||||
|
||||
void string_get_length_test() {
|
||||
@ -305,31 +303,3 @@ void string_last_position_of_test() {
|
||||
assert(string_last_position_of("abcdef", 'f') == 6);
|
||||
assert(string_last_position_of("abcdef", 'g') == 0);
|
||||
}
|
||||
|
||||
void string_pad_start_test() {
|
||||
string_t result = string_pad_start("hello", "ab", 10);
|
||||
assert(assert_string_equal(result, "ababahello"));
|
||||
free(result);
|
||||
|
||||
result = string_pad_start("hello", "ab", 4);
|
||||
assert(assert_string_equal(result, "hell"));
|
||||
free(result);
|
||||
|
||||
result = string_pad_start("hello", "ab", 5);
|
||||
assert(assert_string_equal(result, "hello"));
|
||||
free(result);
|
||||
|
||||
result = string_pad_start("hello", "ab", 6);
|
||||
assert(assert_string_equal(result, "ahello"));
|
||||
free(result);
|
||||
}
|
||||
|
||||
void string_zero_pad_test() {
|
||||
string_t result = string_zero_pad(1, 2);
|
||||
assert(assert_string_equal(result, "01"));
|
||||
free(result);
|
||||
|
||||
result = string_zero_pad(10, 2);
|
||||
assert(assert_string_equal(result, "10"));
|
||||
free(result);
|
||||
}
|
||||
|
@ -6,6 +6,7 @@
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "libcproject.h"
|
||||
#include "test.h"
|
||||
|
||||
void string_test();
|
||||
|
||||
@ -61,8 +62,4 @@ void string_position_of_test();
|
||||
|
||||
void string_last_position_of_test();
|
||||
|
||||
void string_pad_start_test();
|
||||
|
||||
void string_zero_pad_test();
|
||||
|
||||
#endif
|
||||
|
@ -1,7 +1,7 @@
|
||||
#include "assert.h"
|
||||
#include "test.h"
|
||||
|
||||
bool assert_string_equal(const string_t actual, const string_t expected) {
|
||||
if (!string_equals(actual, expected)) {
|
||||
if (strcmp(expected, actual) != 0) {
|
||||
printf("FAIL: expected = \"%s\" ; actual = \"%s\"\n", expected, actual);
|
||||
return false;
|
||||
}
|
||||
@ -9,7 +9,7 @@ bool assert_string_equal(const string_t actual, const string_t expected) {
|
||||
}
|
||||
|
||||
bool assert_string_not_equal(const string_t actual, const string_t expected) {
|
||||
if (string_equals(actual, expected)) {
|
||||
if (strcmp(expected, actual) == 0) {
|
||||
printf("FAIL: expected = \"%s\" ; actual = \"%s\"\n", expected, actual);
|
||||
return false;
|
||||
}
|
14
test/test.h
Normal file
14
test/test.h
Normal file
@ -0,0 +1,14 @@
|
||||
#ifndef __TEST__
|
||||
#define __TEST__
|
||||
|
||||
#include <stdbool.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "libcproject.h"
|
||||
|
||||
bool assert_string_equal(const string_t actual, const string_t expected);
|
||||
|
||||
bool assert_string_not_equal(const string_t actual, const string_t expected);
|
||||
|
||||
#endif
|
Loading…
x
Reference in New Issue
Block a user