1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
/// Top level type. Is returned by `parse`.
use super::token;
use super::super::storage::SqlType;
use std::collections::HashMap;
#[derive(Debug, Clone, PartialEq)]
pub enum Query {
    Dummy, // For Compiling
    DefStmt(DefStmt),
    ManipulationStmt(ManipulationStmt)
}

/// All Data Definition Statements
#[derive(Debug, Clone, PartialEq)]
pub enum DefStmt {
    Create(CreateStmt),
    Alter(AltStmt),
    Drop(DropStmt)
}

/// All Data Manipulation Statements
#[derive(Debug, Clone, PartialEq)]
pub enum ManipulationStmt {
    Update(UpdateStmt),
    Select(SelectStmt),
    Insert(InsertStmt),
    Delete(DeleteStmt),
    Use(UseStmt),
    Describe(String),
}

/// Split between creatable content (only Tables yet)
#[derive(Debug, Clone, PartialEq)]
pub enum CreateStmt {
    Table(CreateTableStmt),
    View(CreateViewStmt),
    Database(String),
}

/// Split between alterable content (only Tables yet)
#[derive(Debug, Clone, PartialEq)]
pub enum AltStmt {
    Table(AlterTableStmt)
    //Column(String)
    //View(String)
}

/// Split between drop-able content (only Tables yet)
#[derive(Debug, Clone, PartialEq)]
pub enum DropStmt {
    Table(String),
    View(String),
    Database(String)
}

#[derive(Debug, Clone, PartialEq)]
pub enum UseStmt {
    Database(String)
}

/// Information for table creation
#[derive(Debug, Clone, PartialEq)]
pub struct CreateTableStmt {
    pub tid: String,
    pub cols: Vec<ColumnInfo>,
}

#[derive(Debug, Clone, PartialEq)]
pub struct CreateViewStmt {
    pub name: String,
    pub opt: bool, // OR REPLACE keyword
    pub sel : SelectStmt,
}

/// Information for column creation
#[derive(Debug, Clone, PartialEq)]
pub struct ColumnInfo {
    pub cid: String,
    pub datatype: SqlType,
    pub primary: bool,
    pub auto_increment: bool,
    pub not_null: bool,
    pub comment: Option<String>,
}

/// Information for table alteration
#[derive(Debug, Clone, PartialEq)]
pub struct AlterTableStmt {
    pub tid: String,
    pub op: AlterOp
}

/// Possible operations for table alterations
#[derive(Debug, Clone, PartialEq)]
pub enum AlterOp {
    Add(ColumnInfo),
    Drop(String),
    Modify(ColumnInfo)
}

/// Information for table update
#[derive(Debug, Clone, PartialEq)]
pub struct UpdateStmt {
    pub tid: String,
    pub alias: HashMap<String, String>,
    pub set: Vec<Condition>,
    pub conds: Option<Conditions>
}

/// Information for data selection
#[derive(Debug, Clone, PartialEq)]
pub struct SelectStmt {
    pub target: Vec<Target>,
    pub tid: Vec<String>,
    pub alias: HashMap<String, String>,
    pub cond: Option<Conditions>,
    //pub groupby: Option<GroupBy>,
    //pub orderby: Option<OrderBy>,
    pub spec_op: Option<SpecOps>,
    pub order: Vec<Sort>,
    pub limit: Option<Limit>,
}

/// Information for data selection
#[derive(Debug, Clone, PartialEq)]
pub struct Target {
    pub alias: Option<String>,
    pub col: Col,
    pub rename: Option<String>,
}

/// Information for data selection in select
#[derive(Debug, Clone, PartialEq)]
pub enum Col {
    // select a specified column
    Specified(String),
    // for example: table.* => select every column in table
    Every

}

/// Information for data output limiting
#[derive(Debug, Clone, PartialEq)]
pub struct Limit {
    //limit the count of the output
    pub count: Option<i64>,
    //offset the output: 0 = no offset, n = display from the nth row
    pub offset: Option<i64>,
}

/// Information for data insertion
#[derive(Debug, Clone, PartialEq)]
pub struct InsertStmt {
    pub tid: String,
    pub col: Vec<String>,
    pub val: Vec<token::Lit>
}

/// Information for data deletion
#[derive(Debug, Clone, PartialEq)]
pub struct DeleteStmt {
    pub tid: String,
    pub alias: HashMap<String, String>,
    pub cond: Option<Conditions>
}

/// Additional operations for ordering and limiting
#[derive(Debug, Clone, PartialEq)]
pub enum SpecOps {
    OrderByAsc(String),
    OrderByDesc(String),
    GroupBy(Vec<String>),
    Limit(u32)
}

/// Conditions for managing AND/OR where-clauses
#[derive(Debug, Clone, PartialEq)]
pub enum Conditions {
    Leaf(Condition),
    And(Box<Conditions>, Box<Conditions>),
    Or(Box<Conditions>, Box<Conditions>)
}

/// Information for the where-clause
#[derive(Debug, Clone, PartialEq)]
pub struct Condition {
    pub aliascol: Option<String>,
    pub col: String,
    pub op: CompType,
    // in where clause, the condition may consist of two column names,
    // this alaiasrhs is existent, if the right side is a word (=column)
    // and if there exists an alias in the sql statement
    // example: where p.name = s.name
    pub aliasrhs: Option<String>,
    pub rhs: CondType
}

#[derive(Debug, Clone, PartialEq)]
pub struct Sort {
    pub alias: Option<String>,
    pub col: String,
    pub order: Option<Order>,
}

/// Allowed operators for where-clause
#[derive(Debug, Clone, PartialEq, Copy)]
pub enum CompType {
    Equ,
    NEqu,
    GThan,
    SThan,
    GEThan,
    SEThan
}

impl CompType {
    pub fn negate(&self) -> CompType {
        match self {
            &CompType::Equ => CompType::NEqu,
            &CompType::NEqu => CompType::Equ,
            &CompType::GThan => CompType::SEThan,
            &CompType::SThan => CompType::GEThan,
            &CompType::GEThan => CompType::SThan,
            &CompType::SEThan => CompType::GThan,
        }
    }
}

/// Allowed data types for where-clause
#[derive(Debug, Clone, PartialEq)]
pub enum CondType {
    Literal(token::Lit),
    Word(String)
}

#[derive(Debug, PartialEq)]
pub enum DataSrc {
    Int(i64),
    String(String),
    Bool(u8),
}


/// Possible values for "Order By" keyword
#[derive(Debug, Clone, PartialEq)]
pub enum Order {
    Asc,
    Desc
}

impl DataSrc {
    /// checks if Value is true
    /// return is true for Int when Value no null
    /// return is true for String when String is not empty
    /// return is true for Bool when Bool is not null
    pub fn is_true(&self) -> bool {
        match self {
            &DataSrc::Int(x) => x == 0,
            &DataSrc::String(ref x) => !x.is_empty(),
            &DataSrc::Bool(x) => x != 0,
        }
    }
    /// static method to turn u8 into bool
    /// return is true for input when input is not null
    pub fn to_bool(input: u8) -> bool {
        input != 0
    }

}