2018-02-11 18:14:34 +01:00
|
|
|
#ifndef EASYCSV_H
|
|
|
|
#define EASYCSV_H
|
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/sendfile.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
/*** Generic type definitions for internal use ***/
|
|
|
|
|
|
|
|
typedef uint32_t row_t; // max length of 2^32 - 1
|
|
|
|
typedef uint32_t column_t;
|
|
|
|
typedef char* rowstr; // non-const, to free it after usage
|
|
|
|
|
|
|
|
/*** Flags ***/
|
|
|
|
|
|
|
|
/** Flags to facilitate changing modes during file copying phases */
|
|
|
|
typedef enum {
|
|
|
|
EASYCSV_UNKNOWNMODE = 0,
|
|
|
|
EASYCSV_R,
|
|
|
|
EASYCSV_W
|
|
|
|
} EASYCSV_MODE;
|
|
|
|
|
|
|
|
/** Flags to parameterise the easycsv_insertvalue */
|
|
|
|
typedef enum {
|
|
|
|
EASYCSV_REPLACE, ///< (default) replaces word
|
|
|
|
EASYCSV_CONCAT, ///< concantates word at the end
|
|
|
|
EASYCSV_RCONCAT, ///< concantates word from the start
|
|
|
|
} EASYCSV_VALMODE;
|
|
|
|
|
|
|
|
/** Flags to parameterise sorting */
|
|
|
|
typedef enum {
|
|
|
|
EASYCSV_ALPHA, ///< sort alphabetically
|
|
|
|
EASYCSV_RALPHA, ///< sort reverse alphabetically
|
|
|
|
EASYCSV_NUMER, ///< sort numerically, from lowest value to highest
|
|
|
|
EASYCSV_RNUMER ///< sort from highest value to lowest
|
|
|
|
} EASYCSV_SORT;
|
|
|
|
|
|
|
|
/** Flags denoting cell type */
|
|
|
|
typedef enum {
|
|
|
|
EASYCSV_NONE,
|
|
|
|
EASYCSV_STRING,
|
|
|
|
EASYCSV_INT,
|
|
|
|
EASYCSV_FLOAT,
|
|
|
|
EASYCSV_UNKNOWNTYPE
|
|
|
|
} _EASYCSV_TYPE;
|
|
|
|
|
|
|
|
/*** Structures ***/
|
2018-02-21 20:40:51 +01:00
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/**
|
|
|
|
* easycsv:
|
|
|
|
*
|
|
|
|
* A CSV file.
|
|
|
|
*/
|
|
|
|
typedef struct _easycsv easycsv;
|
|
|
|
struct _easycsv {
|
|
|
|
FILE *file; ///< original CSV file
|
|
|
|
FILE *temp; ///< temporary CSV file for writing
|
2021-07-17 15:11:57 +02:00
|
|
|
char fp[BUFSIZ];
|
|
|
|
// char *tmpfp;
|
2021-07-16 18:46:36 +02:00
|
|
|
unsigned int rows;
|
|
|
|
unsigned int cols;
|
|
|
|
// EASYCSV_ERRORMSG error;
|
|
|
|
EASYCSV_MODE mode;
|
|
|
|
};
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/*** Constructors ***/
|
2018-02-21 20:40:51 +01:00
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/**
|
|
|
|
* Initialise easycsv
|
|
|
|
* @param[in] relative path to CSV file
|
|
|
|
* @param[in] easycsv mode
|
|
|
|
* @return pointer to easycsv structure
|
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
easycsv*
|
|
|
|
easycsv_init(const char*,
|
|
|
|
const EASYCSV_MODE);
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/* /\** */
|
|
|
|
/* * Initialise easycsv structure with a given error message mode */
|
|
|
|
/* * @param[in] relative path to CSV file */
|
|
|
|
/* * @param[in] easycsv mode */
|
|
|
|
/* * @param[in] easycsv error message mode */
|
|
|
|
/* * @return pointer to easycsv structure */
|
|
|
|
/* *\/ */
|
|
|
|
/* easycsv* */
|
|
|
|
/* easycsv_init_errormsg(const char*, */
|
|
|
|
/* const EASYCSV_MODE, */
|
|
|
|
/* const EASYCSV_ERRORMSG); */
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/**
|
|
|
|
* Destroy easycsv structure
|
|
|
|
* @param[in] Pointer to easycsv stucture
|
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
void
|
|
|
|
easycsv_free(easycsv*);
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/*** Error handling ***/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get error message
|
|
|
|
* @return easycsv error message
|
|
|
|
*/
|
|
|
|
char*
|
|
|
|
easycsv_get_error();
|
|
|
|
|
2021-07-11 17:31:01 +02:00
|
|
|
/*** Access, find ***/
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/**
|
|
|
|
* Find value and returns row and column
|
|
|
|
* @param[in] constant pointer to easycsv structure
|
|
|
|
* @param[in] value to find
|
2021-07-16 21:30:53 +02:00
|
|
|
* @param[out] pointer to column number, unchanged if error
|
|
|
|
* @param[out] pointer to row number, unchanged if error
|
|
|
|
* @return 0 if success, -1 if error
|
2021-07-11 16:44:11 +02:00
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
int
|
2021-07-16 18:46:36 +02:00
|
|
|
easycsv_find_value(const easycsv*,
|
2021-07-16 21:30:53 +02:00
|
|
|
const char*,
|
|
|
|
unsigned int*,
|
|
|
|
unsigned int*);
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/**
|
|
|
|
* Find number of instances of value
|
|
|
|
* @param[in] constant pointer to easycsv structure
|
|
|
|
* @param[in] value to find
|
2021-07-16 21:34:25 +02:00
|
|
|
* @return positive integer if success, -1 if error
|
2021-07-11 16:44:11 +02:00
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
int
|
2021-07-16 18:46:36 +02:00
|
|
|
easycsv_find_num_value(const easycsv*,
|
2021-07-16 21:34:25 +02:00
|
|
|
const char*);
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 17:31:01 +02:00
|
|
|
/*** Access, read ***/
|
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/**
|
|
|
|
* Read string in a specific cell
|
2021-07-16 20:24:23 +02:00
|
|
|
* String is heap-allocated, it must be destroyed (free'd) manually
|
2021-07-11 16:44:11 +02:00
|
|
|
* @param[in] constant pointer to easycsv structure
|
|
|
|
* @param[in] row number
|
|
|
|
* @param[in] column number
|
2021-07-16 20:24:23 +02:00
|
|
|
* @return string value of cell, NULL if error
|
2021-07-11 16:44:11 +02:00
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
char*
|
2021-07-16 18:46:36 +02:00
|
|
|
easycsv_read_value(const easycsv*,
|
2021-07-11 16:44:11 +02:00
|
|
|
unsigned int,
|
|
|
|
unsigned int);
|
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/* /\** */
|
|
|
|
/* * Read string in a specific cell with named column in row 1 */
|
|
|
|
/* * @param[in] constant pointer to easycsv structure */
|
|
|
|
/* * @param[in] row value */
|
|
|
|
/* * @param[in] column number */
|
|
|
|
/* * @return string value of cell, NULL if empty cell */
|
|
|
|
/* *\/ */
|
|
|
|
/* char* */
|
|
|
|
/* easycsv_readcolumnvalue(const easycsv*, */
|
|
|
|
/* const char*, */
|
|
|
|
/* unsigned int); */
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 17:31:01 +02:00
|
|
|
/*** Access, print ***/
|
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/**
|
|
|
|
* Return number of rows
|
|
|
|
* @param[in] constant pointer to easycsv structure
|
2021-07-17 13:51:39 +02:00
|
|
|
* @return number of rows, -1 if error
|
2021-07-11 16:44:11 +02:00
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
int
|
2021-07-16 18:46:36 +02:00
|
|
|
easycsv_print_rows(const easycsv*);
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/**
|
|
|
|
* Return number of columns
|
|
|
|
* @param[in] constant pointer to easycsv structure
|
2021-07-17 13:51:39 +02:00
|
|
|
* @return number of columns, -1 if error
|
2021-07-11 16:44:11 +02:00
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
int
|
2021-07-16 18:46:36 +02:00
|
|
|
easycsv_print_columns(const easycsv*);
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 16:44:11 +02:00
|
|
|
/*** Modifications, sort ***/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sort row
|
|
|
|
* @param[in] pointer to easycsv structure
|
|
|
|
* @param[in] row value
|
|
|
|
* @param[in] easycsv sort mode
|
|
|
|
* @return easycsv error code
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
easycsv_sortrow(easycsv*,
|
|
|
|
const char*,
|
2021-07-11 17:05:18 +02:00
|
|
|
EASYCSV_SORT);
|
2021-07-11 16:44:11 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Sort column
|
|
|
|
* @param[in] pointer to easycsv structure
|
|
|
|
* @param[in] column value
|
|
|
|
* @param[in] easycsv sort mode
|
|
|
|
* @return easycsv error code
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
easycsv_sortcolumn(easycsv*,
|
|
|
|
const char*,
|
2021-07-11 17:05:18 +02:00
|
|
|
EASYCSV_SORT);
|
2021-07-11 16:44:11 +02:00
|
|
|
|
|
|
|
/*** Modifications, insert ***/
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/* /\** */
|
|
|
|
/* * Insert row into CSV file */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @param[in] cell value */
|
|
|
|
/* * @param[in] row number */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_insertrow(easycsv*, */
|
|
|
|
/* const char*, */
|
|
|
|
/* int); */
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 17:05:18 +02:00
|
|
|
/**
|
2021-07-17 15:37:27 +02:00
|
|
|
* Insert/replace string in a specific cell
|
2021-07-11 17:05:18 +02:00
|
|
|
* @param[in] pointer to easycsv structure
|
2021-07-17 15:37:27 +02:00
|
|
|
* @param[in] cell value (cell value will not be modified if there is an error)
|
2021-07-11 17:05:18 +02:00
|
|
|
* @param[in] column number
|
2021-07-17 15:37:27 +02:00
|
|
|
* @param[in] row number
|
|
|
|
* @return 0 if no error, -1 if error
|
2021-07-11 17:05:18 +02:00
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
int
|
|
|
|
easycsv_insertvalue(easycsv*,
|
|
|
|
const char*,
|
2021-07-11 17:05:18 +02:00
|
|
|
unsigned int,
|
|
|
|
unsigned int);
|
2018-02-21 20:40:51 +01:00
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/* /\** */
|
|
|
|
/* * Insert string in a specific cell with specific mode */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @param[in] cell value */
|
|
|
|
/* * @param[in] row number */
|
|
|
|
/* * @param[in] column number */
|
|
|
|
/* * @param[in] easycsv value mode */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_insertvaluemode(easycsv*, */
|
|
|
|
/* const char*, */
|
|
|
|
/* unsigned int, */
|
|
|
|
/* unsigned int, */
|
|
|
|
/* EASYCSV_VALMODE); */
|
|
|
|
|
|
|
|
/* /\** */
|
|
|
|
/* * Insert string in a specific cell with named column in a given row */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @param[in] cell value */
|
|
|
|
/* * @param[in] row number */
|
|
|
|
/* * @param[in] column value */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_insertcolumnvalue(easycsv*, */
|
|
|
|
/* const char*, */
|
|
|
|
/* unsigned int, */
|
|
|
|
/* const char*); */
|
|
|
|
|
|
|
|
/* /\** */
|
|
|
|
/* * Insert string in a specific cell with named column in a given row with specific mode */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @param[in] cell value */
|
|
|
|
/* * @param[in] row number */
|
|
|
|
/* * @param[in] column value */
|
|
|
|
/* * @param[in] easycsv value mode */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_insertcolumnvaluemode(easycsv*, */
|
|
|
|
/* const char*, */
|
|
|
|
/* unsigned int, */
|
|
|
|
/* const char*, */
|
|
|
|
/* EASYCSV_VALMODE); */
|
2018-02-21 20:40:51 +01:00
|
|
|
|
2021-07-11 17:31:01 +02:00
|
|
|
/*** Modifications, push ***/
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/* /\** */
|
|
|
|
/* * Create new column on the right-hand side of an existing one */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @param[in] column value */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_pushcolumn(easycsv*, */
|
|
|
|
/* const char*); */
|
|
|
|
|
|
|
|
/* /\** */
|
|
|
|
/* * Insert string in a vacant cell under a named column */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @param[in] column value */
|
|
|
|
/* * @param[in] cell value */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_pushcolumnvalue(easycsv*, */
|
|
|
|
/* const char*, */
|
|
|
|
/* const char*); */
|
2018-02-21 20:40:51 +01:00
|
|
|
|
2021-07-11 17:31:01 +02:00
|
|
|
/*** Modifications, delete ***/
|
2018-02-21 20:40:51 +01:00
|
|
|
|
2021-07-11 17:31:01 +02:00
|
|
|
/**
|
|
|
|
* Delete cell value
|
|
|
|
* @param[in] pointer to easycsv structure
|
|
|
|
* @param[in] row number
|
|
|
|
* @param[in] column number
|
|
|
|
* @return easycsv error code
|
|
|
|
*/
|
2018-02-21 20:40:51 +01:00
|
|
|
int
|
|
|
|
easycsv_deletevalue(easycsv*,
|
2021-07-11 17:31:01 +02:00
|
|
|
unsigned int,
|
|
|
|
unsigned int);
|
2018-02-21 20:40:51 +01:00
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/* /\** */
|
|
|
|
/* * Delete numbered column */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @param[in] column number */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_deletecolumnint(easycsv*, */
|
|
|
|
/* unsigned int); */
|
|
|
|
|
|
|
|
/* /\** Delete named column */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @param[in] column value */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_deletecolumnstr(easycsv*, */
|
|
|
|
/* const char*); */
|
2018-02-21 20:40:51 +01:00
|
|
|
|
2018-02-22 14:23:23 +01:00
|
|
|
/* Delete numered row */
|
|
|
|
|
|
|
|
/* Delete named row */
|
2018-02-11 18:14:34 +01:00
|
|
|
|
2021-07-11 17:31:01 +02:00
|
|
|
/*** Modifications, append ***/
|
2021-07-11 16:44:11 +02:00
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/* /\** */
|
|
|
|
/* * Append two CSV files */
|
|
|
|
/* * @param[in] destination CSV file */
|
|
|
|
/* * @param[in] source CSV file */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_appendcsv(easycsv*, */
|
|
|
|
/* easycsv*); */
|
2021-07-11 16:44:11 +02:00
|
|
|
|
|
|
|
/*** Modifications, miscellenaeous ***/
|
|
|
|
|
2021-07-16 18:46:36 +02:00
|
|
|
/* /\** */
|
|
|
|
/* * Compress a CSV file */
|
|
|
|
/* * @param[in] pointer to easycsv structure */
|
|
|
|
/* * @return easycsv error code */
|
|
|
|
/* *\/ */
|
|
|
|
/* int */
|
|
|
|
/* easycsv_compress(easycsv*); */
|
2021-07-11 16:44:11 +02:00
|
|
|
|
2018-02-11 18:14:34 +01:00
|
|
|
#endif /* EASYCSV_H */
|