Add files via upload
This commit is contained in:
parent
4d82bb82a8
commit
dc95d79945
25
Makefile
Normal file
25
Makefile
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
CC = gcc
|
||||||
|
CFLAGS = -Wall -Wextra -Iinclude
|
||||||
|
|
||||||
|
SRC_DIR = src
|
||||||
|
BUILD_DIR = build
|
||||||
|
BIN_DIR = bin
|
||||||
|
|
||||||
|
SRC_FILES = $(wildcard $(SRC_DIR)/*.c)
|
||||||
|
OBJ_FILES = $(patsubst $(SRC_DIR)/%.c, $(BUILD_DIR)/%.o, $(SRC_FILES))
|
||||||
|
TARGET = $(BIN_DIR)/main
|
||||||
|
|
||||||
|
all: $(TARGET)
|
||||||
|
|
||||||
|
$(TARGET): $(OBJ_FILES)
|
||||||
|
@mkdir -p $(BIN_DIR)
|
||||||
|
$(CC) $(OBJ_FILES) -o $@
|
||||||
|
|
||||||
|
$(BUILD_DIR)/%.o: $(SRC_DIR)/%.c
|
||||||
|
@mkdir -p $(BUILD_DIR)
|
||||||
|
$(CC) $(CFLAGS) -c $< -o $@
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -rf $(BUILD_DIR) $(BIN_DIR)
|
||||||
|
|
||||||
|
.PHONY: all clean
|
49
README.md
49
README.md
@ -1,2 +1,47 @@
|
|||||||
# SimpleDB
|
## Simple DB
|
||||||
SimpleDB
|
|
||||||
|
Simple DB proof of concept that utilizes SQL syntax for REPL.
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
**Loading DB from file**
|
||||||
|
|
||||||
|
```
|
||||||
|
Simple SQL-like Database
|
||||||
|
Copyright (c) 2025 Ivan Nikolskiy, All Rights Reserved.
|
||||||
|
|
||||||
|
Supported commands: CREATE TABLE, INSERT INTO, SELECT * FROM, SAVE, LOAD
|
||||||
|
|
||||||
|
Enter SQL query: LOAD
|
||||||
|
Database loaded from 'database.db'.
|
||||||
|
Enter SQL query: SELECT * FROM Students
|
||||||
|
Table: Students
|
||||||
|
Name Age Major
|
||||||
|
Alice 20 CS
|
||||||
|
Bob 20 CS
|
||||||
|
Enter SQL query: EXIT
|
||||||
|
```
|
||||||
|
|
||||||
|
**Writing your DB to file**
|
||||||
|
|
||||||
|
```
|
||||||
|
Simple SQL-like Database
|
||||||
|
Copyright (c) 2025 Ivan Nikolskiy, All Rights Reserved.
|
||||||
|
|
||||||
|
Supported commands: CREATE TABLE, INSERT INTO, SELECT * FROM, SAVE, LOAD
|
||||||
|
|
||||||
|
Enter SQL query: CREATE TABLE Students (Name, Age, Major)
|
||||||
|
Table 'Students' with 3 columns created successfully.
|
||||||
|
Enter SQL query: INSERT INTO Students VALUES (Alice, 20, CS)
|
||||||
|
Row inserted into table 'Students'.
|
||||||
|
Enter SQL query: INSERT INTO Students VALUES (Bob, 20, CS)
|
||||||
|
Row inserted into table 'Students'.
|
||||||
|
Enter SQL query: SELECT * FROM Students
|
||||||
|
Table: Students
|
||||||
|
Name Age Major
|
||||||
|
Alice 20 CS
|
||||||
|
Bob 20 CS
|
||||||
|
Enter SQL query: SAVE
|
||||||
|
Database saved to 'database.db'.
|
||||||
|
Enter SQL query: EXIT
|
||||||
|
```
|
||||||
|
BIN
database.db
Normal file
BIN
database.db
Normal file
Binary file not shown.
43
include/db.h
Normal file
43
include/db.h
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
#ifndef _DB_H_
|
||||||
|
#define _DB_H_
|
||||||
|
|
||||||
|
#define MAX_TABLES 10
|
||||||
|
#define MAX_COLUMNS 10
|
||||||
|
#define MAX_ROWS 100
|
||||||
|
#define MAX_QUERY_LENGTH 256
|
||||||
|
#define MAX_NAME_LENGTH 50
|
||||||
|
#define DB_FILE "database.db"
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
char name[MAX_NAME_LENGTH];
|
||||||
|
char data[MAX_ROWS][MAX_NAME_LENGTH];
|
||||||
|
} Column;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
char name[MAX_NAME_LENGTH];
|
||||||
|
Column columns[MAX_COLUMNS];
|
||||||
|
int column_count;
|
||||||
|
int row_count;
|
||||||
|
} Table;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
Table tables[MAX_TABLES];
|
||||||
|
int table_count;
|
||||||
|
} Database;
|
||||||
|
|
||||||
|
Database db;
|
||||||
|
|
||||||
|
void create_table(const char *table_name, char *columns);
|
||||||
|
void insert_into_table(const char *table_name, char *values);
|
||||||
|
void select_from_table(const char *table_name);
|
||||||
|
void parse_query(const char *query);
|
||||||
|
Table *find_table(const char *table_name);
|
||||||
|
void save_database_to_file(const char *filename);
|
||||||
|
void load_database_from_file(const char *filename);
|
||||||
|
int validate_ipv4_address(const char *ip);
|
||||||
|
void trim_whitespace(char *str);
|
||||||
|
|
||||||
|
#endif
|
113
include/linenoise.h
Normal file
113
include/linenoise.h
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
/* linenoise.h -- VERSION 1.0
|
||||||
|
*
|
||||||
|
* Guerrilla line editing library against the idea that a line editing lib
|
||||||
|
* needs to be 20,000 lines of C code.
|
||||||
|
*
|
||||||
|
* See linenoise.c for more information.
|
||||||
|
*
|
||||||
|
* ------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* Copyright (c) 2010-2023, Salvatore Sanfilippo <antirez at gmail dot com>
|
||||||
|
* Copyright (c) 2010-2013, Pieter Noordhuis <pcnoordhuis at gmail dot com>
|
||||||
|
*
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are
|
||||||
|
* met:
|
||||||
|
*
|
||||||
|
* * Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* * Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __LINENOISE_H
|
||||||
|
#define __LINENOISE_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stddef.h> /* For size_t. */
|
||||||
|
|
||||||
|
extern char *linenoiseEditMore;
|
||||||
|
|
||||||
|
/* The linenoiseState structure represents the state during line editing.
|
||||||
|
* We pass this state to functions implementing specific editing
|
||||||
|
* functionalities. */
|
||||||
|
struct linenoiseState {
|
||||||
|
int in_completion; /* The user pressed TAB and we are now in completion
|
||||||
|
* mode, so input is handled by completeLine(). */
|
||||||
|
size_t completion_idx; /* Index of next completion to propose. */
|
||||||
|
int ifd; /* Terminal stdin file descriptor. */
|
||||||
|
int ofd; /* Terminal stdout file descriptor. */
|
||||||
|
char *buf; /* Edited line buffer. */
|
||||||
|
size_t buflen; /* Edited line buffer size. */
|
||||||
|
const char *prompt; /* Prompt to display. */
|
||||||
|
size_t plen; /* Prompt length. */
|
||||||
|
size_t pos; /* Current cursor position. */
|
||||||
|
size_t oldpos; /* Previous refresh cursor position. */
|
||||||
|
size_t len; /* Current edited line length. */
|
||||||
|
size_t cols; /* Number of columns in terminal. */
|
||||||
|
size_t oldrows; /* Rows used by last refrehsed line (multiline mode) */
|
||||||
|
int history_index; /* The history index we are currently editing. */
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct linenoiseCompletions {
|
||||||
|
size_t len;
|
||||||
|
char **cvec;
|
||||||
|
} linenoiseCompletions;
|
||||||
|
|
||||||
|
/* Non blocking API. */
|
||||||
|
int linenoiseEditStart(struct linenoiseState *l, int stdin_fd, int stdout_fd, char *buf, size_t buflen, const char *prompt);
|
||||||
|
char *linenoiseEditFeed(struct linenoiseState *l);
|
||||||
|
void linenoiseEditStop(struct linenoiseState *l);
|
||||||
|
void linenoiseHide(struct linenoiseState *l);
|
||||||
|
void linenoiseShow(struct linenoiseState *l);
|
||||||
|
|
||||||
|
/* Blocking API. */
|
||||||
|
char *linenoise(const char *prompt);
|
||||||
|
void linenoiseFree(void *ptr);
|
||||||
|
|
||||||
|
/* Completion API. */
|
||||||
|
typedef void(linenoiseCompletionCallback)(const char *, linenoiseCompletions *);
|
||||||
|
typedef char*(linenoiseHintsCallback)(const char *, int *color, int *bold);
|
||||||
|
typedef void(linenoiseFreeHintsCallback)(void *);
|
||||||
|
void linenoiseSetCompletionCallback(linenoiseCompletionCallback *);
|
||||||
|
void linenoiseSetHintsCallback(linenoiseHintsCallback *);
|
||||||
|
void linenoiseSetFreeHintsCallback(linenoiseFreeHintsCallback *);
|
||||||
|
void linenoiseAddCompletion(linenoiseCompletions *, const char *);
|
||||||
|
|
||||||
|
/* History API. */
|
||||||
|
int linenoiseHistoryAdd(const char *line);
|
||||||
|
int linenoiseHistorySetMaxLen(int len);
|
||||||
|
int linenoiseHistorySave(const char *filename);
|
||||||
|
int linenoiseHistoryLoad(const char *filename);
|
||||||
|
|
||||||
|
/* Other utilities. */
|
||||||
|
void linenoiseClearScreen(void);
|
||||||
|
void linenoiseSetMultiLine(int ml);
|
||||||
|
void linenoisePrintKeyCodes(void);
|
||||||
|
void linenoiseMaskModeEnable(void);
|
||||||
|
void linenoiseMaskModeDisable(void);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* __LINENOISE_H */
|
376
src/db.c
Normal file
376
src/db.c
Normal file
@ -0,0 +1,376 @@
|
|||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
|
||||||
|
#include "db.h"
|
||||||
|
|
||||||
|
void trim_whitespace(char *str)
|
||||||
|
{
|
||||||
|
char *end;
|
||||||
|
|
||||||
|
while (isspace((unsigned char)*str))
|
||||||
|
{
|
||||||
|
str++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*str == 0)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
end = str + strlen(str) - 1;
|
||||||
|
|
||||||
|
while (end > str && isspace((unsigned char)*end))
|
||||||
|
{
|
||||||
|
end--;
|
||||||
|
}
|
||||||
|
|
||||||
|
end[1] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
int validate_ipv4_address(const char *ip)
|
||||||
|
{
|
||||||
|
int segments;
|
||||||
|
int ch_count;
|
||||||
|
const char *ptr;
|
||||||
|
|
||||||
|
segments = 0;
|
||||||
|
ch_count = 0;
|
||||||
|
ptr = ip;
|
||||||
|
|
||||||
|
while (*ptr)
|
||||||
|
{
|
||||||
|
if (*ptr == '.')
|
||||||
|
{
|
||||||
|
if (ch_count == 0 || ch_count > 3)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
segments++;
|
||||||
|
ch_count = 0;
|
||||||
|
}
|
||||||
|
else if (!isdigit((unsigned char)*ptr))
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ch_count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
ptr++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (segments != 3 || ch_count == 0 || ch_count > 3)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Table *find_table(const char *table_name)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < db.table_count; i++)
|
||||||
|
{
|
||||||
|
if (strcmp(db.tables[i].name, table_name) == 0)
|
||||||
|
{
|
||||||
|
return &db.tables[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void create_table(const char *table_name, char *columns)
|
||||||
|
{
|
||||||
|
Table *table;
|
||||||
|
char *token;
|
||||||
|
|
||||||
|
if (db.table_count >= MAX_TABLES)
|
||||||
|
{
|
||||||
|
printf("Error: Maximum table limit reached.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (find_table(table_name))
|
||||||
|
{
|
||||||
|
printf("Error: Table '%s' already exists.\n", table_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
table = &db.tables[db.table_count++];
|
||||||
|
strcpy(table->name, table_name);
|
||||||
|
table->column_count = 0;
|
||||||
|
table->row_count = 0;
|
||||||
|
|
||||||
|
token = strtok(columns, ",");
|
||||||
|
|
||||||
|
while (token && table->column_count < MAX_COLUMNS)
|
||||||
|
{
|
||||||
|
trim_whitespace(token);
|
||||||
|
strcpy(table->columns[table->column_count++].name, token);
|
||||||
|
token = strtok(NULL, ",");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (table->column_count == 0)
|
||||||
|
{
|
||||||
|
printf("Error: No columns defined for table '%s'.\n", table_name);
|
||||||
|
db.table_count--;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("Table '%s' with %d columns created successfully.\n", table_name, table->column_count);
|
||||||
|
}
|
||||||
|
|
||||||
|
void insert_into_table(const char *table_name, char *values)
|
||||||
|
{
|
||||||
|
Table *table;
|
||||||
|
char *token;
|
||||||
|
int col_index;
|
||||||
|
|
||||||
|
table = find_table(table_name);
|
||||||
|
|
||||||
|
if (!table)
|
||||||
|
{
|
||||||
|
printf("Error: Table '%s' does not exist.\n", table_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (table->column_count == 0)
|
||||||
|
{
|
||||||
|
printf("Error: Table '%s' has no columns defined.\n", table_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (table->row_count >= MAX_ROWS)
|
||||||
|
{
|
||||||
|
printf("Error: Maximum row limit reached for table '%s'.\n", table_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
token = strtok(values, ",");
|
||||||
|
col_index = 0;
|
||||||
|
|
||||||
|
while (token && col_index < table->column_count)
|
||||||
|
{
|
||||||
|
trim_whitespace(token);
|
||||||
|
|
||||||
|
if (strcmp(table->columns[col_index].name, "IPv4") == 0)
|
||||||
|
{
|
||||||
|
if (!validate_ipv4_address(token))
|
||||||
|
{
|
||||||
|
printf("Error: Invalid IPv4 address '%s'.\n", token);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
strcpy(table->columns[col_index].data[table->row_count], token);
|
||||||
|
token = strtok(NULL, ",");
|
||||||
|
col_index++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (col_index != table->column_count)
|
||||||
|
{
|
||||||
|
printf("Error: Column count mismatch for table '%s'.\n", table_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
table->row_count++;
|
||||||
|
printf("Row inserted into table '%s'.\n", table_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void select_from_table(const char *table_name)
|
||||||
|
{
|
||||||
|
Table *table;
|
||||||
|
int i;
|
||||||
|
int j;
|
||||||
|
|
||||||
|
table = find_table(table_name);
|
||||||
|
|
||||||
|
if (!table)
|
||||||
|
{
|
||||||
|
printf("Error: Table '%s' does not exist.\n", table_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("Table: %s\n", table->name);
|
||||||
|
|
||||||
|
for (i = 0; i < table->column_count; i++)
|
||||||
|
{
|
||||||
|
printf("%s\t", table->columns[i].name);
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("\n");
|
||||||
|
|
||||||
|
for (i = 0; i < table->row_count; i++)
|
||||||
|
{
|
||||||
|
for (j = 0; j < table->column_count; j++)
|
||||||
|
{
|
||||||
|
printf("%s\t", table->columns[j].data[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void save_database_to_file(const char *filename)
|
||||||
|
{
|
||||||
|
FILE *file;
|
||||||
|
|
||||||
|
file = fopen(filename, "wb");
|
||||||
|
|
||||||
|
if (!file)
|
||||||
|
{
|
||||||
|
printf("Error: Could not open file '%s' for writing.\n", filename);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fwrite(&db, sizeof(Database), 1, file);
|
||||||
|
|
||||||
|
fclose(file);
|
||||||
|
printf("Database saved to '%s'.\n", filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
void load_database_from_file(const char *filename)
|
||||||
|
{
|
||||||
|
FILE *file;
|
||||||
|
|
||||||
|
file = fopen(filename, "rb");
|
||||||
|
|
||||||
|
if (!file)
|
||||||
|
{
|
||||||
|
printf("Error: Could not open file '%s' for reading.\n", filename);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fread(&db, sizeof(Database), 1, file);
|
||||||
|
|
||||||
|
fclose(file);
|
||||||
|
printf("Database loaded from '%s'.\n", filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_query(const char *query)
|
||||||
|
{
|
||||||
|
char query_copy[MAX_QUERY_LENGTH];
|
||||||
|
char *command;
|
||||||
|
char *table_name;
|
||||||
|
char *columns;
|
||||||
|
|
||||||
|
strcpy(query_copy, query);
|
||||||
|
command = strtok(query_copy, " ");
|
||||||
|
|
||||||
|
if (strcmp(command, "CREATE") == 0)
|
||||||
|
{
|
||||||
|
char *next_token = strtok(NULL, " ");
|
||||||
|
if (next_token == NULL || strcmp(next_token, "TABLE") != 0)
|
||||||
|
{
|
||||||
|
printf("Error: Invalid CREATE TABLE syntax.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
table_name = strtok(NULL, " ");
|
||||||
|
if (table_name == NULL)
|
||||||
|
{
|
||||||
|
printf("Error: Table name is missing.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
columns = strchr(query, '(');
|
||||||
|
if (columns == NULL)
|
||||||
|
{
|
||||||
|
printf("Error: Missing column definitions.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
columns++;
|
||||||
|
char *closing_paren = strchr(columns, ')');
|
||||||
|
if (closing_paren == NULL)
|
||||||
|
{
|
||||||
|
printf("Error: Missing closing parenthesis in column definitions.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
*closing_paren = '\0';
|
||||||
|
trim_whitespace(columns);
|
||||||
|
|
||||||
|
if (strlen(columns) == 0)
|
||||||
|
{
|
||||||
|
printf("Error: No columns defined for table '%s'.\n", table_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
create_table(table_name, columns);
|
||||||
|
}
|
||||||
|
else if (strcmp(command, "INSERT") == 0)
|
||||||
|
{
|
||||||
|
char *next_token = strtok(NULL, " ");
|
||||||
|
if (next_token == NULL || strcmp(next_token, "INTO") != 0)
|
||||||
|
{
|
||||||
|
printf("Error: Invalid INSERT INTO syntax.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
table_name = strtok(NULL, " ");
|
||||||
|
if (table_name == NULL)
|
||||||
|
{
|
||||||
|
printf("Error: Table name is missing.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
columns = strchr(query, '(');
|
||||||
|
if (columns == NULL)
|
||||||
|
{
|
||||||
|
printf("Error: Missing values.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
columns++;
|
||||||
|
char *closing_paren = strchr(columns, ')');
|
||||||
|
if (closing_paren == NULL)
|
||||||
|
{
|
||||||
|
printf("Error: Missing closing parenthesis in values.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
*closing_paren = '\0';
|
||||||
|
trim_whitespace(columns);
|
||||||
|
|
||||||
|
if (strlen(columns) == 0)
|
||||||
|
{
|
||||||
|
printf("Error: No values provided for table '%s'.\n", table_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
insert_into_table(table_name, columns);
|
||||||
|
}
|
||||||
|
else if (strcmp(command, "SELECT") == 0)
|
||||||
|
{
|
||||||
|
strtok(NULL, " ");
|
||||||
|
strtok(NULL, " ");
|
||||||
|
table_name = strtok(NULL, " ");
|
||||||
|
if (table_name == NULL)
|
||||||
|
{
|
||||||
|
printf("Error: Table name is missing in SELECT query.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
select_from_table(table_name);
|
||||||
|
}
|
||||||
|
else if (strcmp(command, "SAVE") == 0)
|
||||||
|
{
|
||||||
|
save_database_to_file(DB_FILE);
|
||||||
|
}
|
||||||
|
else if (strcmp(command, "LOAD") == 0)
|
||||||
|
{
|
||||||
|
load_database_from_file(DB_FILE);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
printf("Error: Unsupported query.\n");
|
||||||
|
}
|
||||||
|
}
|
1349
src/linenoise.c
Normal file
1349
src/linenoise.c
Normal file
File diff suppressed because it is too large
Load Diff
44
src/main.c
Normal file
44
src/main.c
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "linenoise.h"
|
||||||
|
#include "db.h"
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
char *query;
|
||||||
|
|
||||||
|
db.table_count = 0;
|
||||||
|
|
||||||
|
printf("Simple SQL-like Database\n");
|
||||||
|
printf("Copyright (c) 2025 Ivan Nikolskiy, All Rights Reserved.\n\n");
|
||||||
|
printf("Supported commands: CREATE TABLE, INSERT INTO, SELECT * FROM, SAVE, LOAD\n\n");
|
||||||
|
|
||||||
|
while (1)
|
||||||
|
{
|
||||||
|
query = linenoise("Enter SQL query: ");
|
||||||
|
if (!query)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
trim_whitespace(query);
|
||||||
|
|
||||||
|
if (strcmp(query, "EXIT") == 0)
|
||||||
|
{
|
||||||
|
free(query);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strlen(query) > 0)
|
||||||
|
{
|
||||||
|
parse_query(query);
|
||||||
|
linenoiseHistoryAdd(query);
|
||||||
|
}
|
||||||
|
|
||||||
|
free(query);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user