Compare commits
2 Commits
4bb9f2813d
...
61de38b9bf
| Author | SHA1 | Date | |
|---|---|---|---|
| 61de38b9bf | |||
| 6c5d3910dc |
@ -11,7 +11,7 @@ pub struct Model {
|
|||||||
pub transaction_type: String,
|
pub transaction_type: String,
|
||||||
pub total_amount: Decimal,
|
pub total_amount: Decimal,
|
||||||
pub timestamp: DateTime,
|
pub timestamp: DateTime,
|
||||||
pub title: String,
|
pub title: Option<String>,
|
||||||
pub emoji: Option<String>,
|
pub emoji: Option<String>,
|
||||||
pub notes: Option<String>,
|
pub notes: Option<String>,
|
||||||
pub receipt: Option<String>,
|
pub receipt: Option<String>,
|
||||||
|
|||||||
3
fixtures/transactions.csv
Normal file
3
fixtures/transactions.csv
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
Transaction ID,Date,Time,Type,Name,Emoji,Category,Amount,Currency,Local amount,Local currency,Notes and #tags,Address,Receipt,Description,Category split
|
||||||
|
tx_0000AhmcbFKVlkAFbeZuAk,11/05/2024,13:41:47,Pot transfer,Savings Pot,,Savings,-1.4,GBP,-1.4,GBP,,,,Round up,
|
||||||
|
tx_0000AhmgdZ0FzkVZou77Ev,11/05/2024,14:27:01,Card payment,D,💊,C,-100.42,GBP,-100.42,GBP,,B,,A,"Groceries:15.00,Personal care:10.00"
|
||||||
|
38
fixtures/transactions.json
Normal file
38
fixtures/transactions.json
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
[
|
||||||
|
[
|
||||||
|
"tx_0000AhmcbFKVlkAFbeZuAk",
|
||||||
|
"2024-05-11T00:00:00.000Z",
|
||||||
|
"1899-12-30T13:41:47.000Z",
|
||||||
|
"Pot transfer",
|
||||||
|
"Savings Pot",
|
||||||
|
"",
|
||||||
|
"Savings",
|
||||||
|
-1.4,
|
||||||
|
"GBP",
|
||||||
|
-1.4,
|
||||||
|
"GBP",
|
||||||
|
"",
|
||||||
|
"",
|
||||||
|
"",
|
||||||
|
"Round up",
|
||||||
|
""
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"tx_0000AhmgdZ0FzkVZou77Ev",
|
||||||
|
"2024-05-11T00:00:00.000Z",
|
||||||
|
"1899-12-30T14:27:01.000Z",
|
||||||
|
"Card payment",
|
||||||
|
"D",
|
||||||
|
"💊",
|
||||||
|
"C",
|
||||||
|
-100.42,
|
||||||
|
"GBP",
|
||||||
|
-100.42,
|
||||||
|
"GBP",
|
||||||
|
"",
|
||||||
|
"B",
|
||||||
|
"",
|
||||||
|
"A",
|
||||||
|
"Groceries:15.00,Personal care:10.00"
|
||||||
|
]
|
||||||
|
]
|
||||||
12
justfile
Executable file
12
justfile
Executable file
@ -0,0 +1,12 @@
|
|||||||
|
#!/usr/bin/env just --justfile
|
||||||
|
|
||||||
|
generate-migration:
|
||||||
|
sea-orm-cli migrate generate $1
|
||||||
|
|
||||||
|
migrate:
|
||||||
|
sea-orm-cli migrate up
|
||||||
|
|
||||||
|
generate-entities:
|
||||||
|
sea-orm-cli generate entity --with-serde both \
|
||||||
|
-l -o entity/src \
|
||||||
|
--ignore-tables monzo_ingestion_seaql_migrations
|
||||||
@ -2,6 +2,7 @@ pub use sea_orm_migration::prelude::*;
|
|||||||
|
|
||||||
pub mod m20230904_141851_create_monzo_tables;
|
pub mod m20230904_141851_create_monzo_tables;
|
||||||
mod m20240529_195030_add_transaction_identity_hash;
|
mod m20240529_195030_add_transaction_identity_hash;
|
||||||
|
mod m20240603_162500_make_title_optional;
|
||||||
|
|
||||||
pub struct Migrator;
|
pub struct Migrator;
|
||||||
|
|
||||||
@ -15,6 +16,7 @@ impl MigratorTrait for Migrator {
|
|||||||
vec![
|
vec![
|
||||||
Box::new(m20230904_141851_create_monzo_tables::Migration),
|
Box::new(m20230904_141851_create_monzo_tables::Migration),
|
||||||
Box::new(m20240529_195030_add_transaction_identity_hash::Migration),
|
Box::new(m20240529_195030_add_transaction_identity_hash::Migration),
|
||||||
|
Box::new(m20240603_162500_make_title_optional::Migration),
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
47
migration/src/m20240603_162500_make_title_optional.rs
Normal file
47
migration/src/m20240603_162500_make_title_optional.rs
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
use sea_orm_migration::prelude::*;
|
||||||
|
|
||||||
|
#[derive(DeriveMigrationName)]
|
||||||
|
pub struct Migration;
|
||||||
|
|
||||||
|
#[async_trait::async_trait]
|
||||||
|
impl MigrationTrait for Migration {
|
||||||
|
async fn up(&self, manager: &SchemaManager) -> Result<(), DbErr> {
|
||||||
|
manager.alter_table(
|
||||||
|
TableAlterStatement::new()
|
||||||
|
.table(Transaction::Table)
|
||||||
|
.modify_column(ColumnDef::new(Transaction::Title).string().null())
|
||||||
|
.to_owned()
|
||||||
|
).await?;
|
||||||
|
|
||||||
|
// Set all empty string titles to null
|
||||||
|
manager.get_connection().execute_unprepared(r#"
|
||||||
|
update transaction
|
||||||
|
set title = null
|
||||||
|
where title = ''
|
||||||
|
"#).await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn down(&self, manager: &SchemaManager) -> Result<(), DbErr> {
|
||||||
|
// Set all null titles to empty string when reverting
|
||||||
|
manager.get_connection().execute_unprepared(r#"
|
||||||
|
update transaction
|
||||||
|
set title = ''
|
||||||
|
where title is null
|
||||||
|
"#).await?;
|
||||||
|
|
||||||
|
manager.alter_table(
|
||||||
|
TableAlterStatement::new()
|
||||||
|
.table(Transaction::Table)
|
||||||
|
.modify_column(ColumnDef::new(Transaction::Title).string().not_null())
|
||||||
|
.to_owned()
|
||||||
|
).await
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(DeriveIden)]
|
||||||
|
enum Transaction {
|
||||||
|
Table,
|
||||||
|
Title,
|
||||||
|
}
|
||||||
@ -40,7 +40,7 @@ pub struct MonzoRow {
|
|||||||
emoji: Option<String>,
|
emoji: Option<String>,
|
||||||
description: Option<String>,
|
description: Option<String>,
|
||||||
transaction_type: String,
|
transaction_type: String,
|
||||||
title: String,
|
title: Option<String>,
|
||||||
timestamp: NaiveDateTime,
|
timestamp: NaiveDateTime,
|
||||||
transaction_id: String,
|
transaction_id: String,
|
||||||
}
|
}
|
||||||
@ -153,7 +153,7 @@ pub fn from_json_row(row: Vec<Value>) -> anyhow::Result<MonzoRow> {
|
|||||||
|
|
||||||
Ok(MonzoRow {
|
Ok(MonzoRow {
|
||||||
transaction_id: json_required_str(&row[headings::TRANSACTION_ID], "Transaction ID")?,
|
transaction_id: json_required_str(&row[headings::TRANSACTION_ID], "Transaction ID")?,
|
||||||
title: json_required_str(&row[headings::NAME], "Title")?,
|
title: json_opt(&row[headings::NAME]),
|
||||||
transaction_type: json_required_str(&row[headings::TYPE], "Transaction type")?,
|
transaction_type: json_required_str(&row[headings::TYPE], "Transaction type")?,
|
||||||
description: json_opt(&row[headings::DESCRIPTION]),
|
description: json_opt(&row[headings::DESCRIPTION]),
|
||||||
emoji: json_opt(&row[headings::EMOJI]),
|
emoji: json_opt(&row[headings::EMOJI]),
|
||||||
@ -166,6 +166,32 @@ pub fn from_json_row(row: Vec<Value>) -> anyhow::Result<MonzoRow> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_json() {
|
||||||
|
let json = include_str!("../../fixtures/transactions.json");
|
||||||
|
let csv = include_str!("../../fixtures/transactions.csv");
|
||||||
|
|
||||||
|
let json: Vec<Vec<Value>> = serde_json::from_str(json).unwrap();
|
||||||
|
let mut csv_reader = csv::Reader::from_reader(csv.as_bytes());
|
||||||
|
|
||||||
|
let json_rows = json.iter()
|
||||||
|
.map(|row| from_json_row(row.clone()))
|
||||||
|
.collect::<Result<Vec<_>, anyhow::Error>>()
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let csv_rows = csv_reader.records()
|
||||||
|
.map(|record| from_csv_row(record.unwrap()))
|
||||||
|
.collect::<Result<Vec<_>, anyhow::Error>>()
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
assert_eq!(csv_rows.len(), json_rows.len(), "Different number of rows");
|
||||||
|
|
||||||
|
for (i, (json_row, csv_row)) in json_rows.iter().zip(csv_rows.iter()).enumerate() {
|
||||||
|
assert_eq!(json_row, csv_row, "Row {} is different", i);
|
||||||
|
assert_eq!(json_row.compute_hash(), csv_row.compute_hash(), "Row {} hash are different", i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn csv_opt(s: &str) -> Option<String> {
|
fn csv_opt(s: &str) -> Option<String> {
|
||||||
match s {
|
match s {
|
||||||
"" => None,
|
"" => None,
|
||||||
@ -185,7 +211,7 @@ pub fn from_csv_row(row: StringRecord) -> anyhow::Result<MonzoRow> {
|
|||||||
Ok(MonzoRow {
|
Ok(MonzoRow {
|
||||||
timestamp,
|
timestamp,
|
||||||
transaction_id: row[headings::TRANSACTION_ID].to_string(),
|
transaction_id: row[headings::TRANSACTION_ID].to_string(),
|
||||||
title: row[headings::NAME].to_string(),
|
title: csv_opt(&row[headings::NAME]),
|
||||||
transaction_type: row[headings::TYPE].to_string(),
|
transaction_type: row[headings::TYPE].to_string(),
|
||||||
description: csv_opt(&row[headings::DESCRIPTION]),
|
description: csv_opt(&row[headings::DESCRIPTION]),
|
||||||
emoji: csv_opt(&row[headings::EMOJI]),
|
emoji: csv_opt(&row[headings::EMOJI]),
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user