1
use std::{
2
    collections::{BTreeMap, HashSet},
3
    ops::RangeInclusive,
4
};
5

            
6
use fake::{
7
    faker::{
8
        address::raw::{
9
            BuildingNumber, CityName, CountryCode, PostCode, StateName, StreetName, StreetSuffix,
10
        },
11
        company::raw::{Bs, BsAdj},
12
        internet::raw::SafeEmail,
13
        lorem::raw::Paragraphs,
14
        name::raw::Name,
15
        phone_number::raw::PhoneNumber,
16
    },
17
    locales::EN,
18
    Dummy, Fake,
19
};
20
use rand::{seq::SliceRandom, Rng};
21
use serde::{Deserialize, Serialize};
22

            
23
use crate::utils::gen_range;
24

            
25
4
#[derive(Default, Clone, Debug)]
26
pub struct InitialDataSet {
27
    pub customers: BTreeMap<u32, Customer>,
28
    pub products: BTreeMap<u32, Product>,
29
    pub categories: BTreeMap<u32, Category>,
30
    pub orders: BTreeMap<u32, Order>,
31
    pub reviews: Vec<ProductReview>,
32
}
33

            
34
pub struct InitialDataSetConfig {
35
    pub number_of_customers: RangeInclusive<u32>,
36
    pub number_of_products: RangeInclusive<u32>,
37
    pub number_of_categories: RangeInclusive<u32>,
38
    pub number_of_orders: RangeInclusive<u32>,
39
    pub number_of_reviews: RangeInclusive<u32>,
40
}
41

            
42
impl InitialDataSetConfig {
43
4
    pub fn fake<R: Rng>(&self, rng: &mut R) -> InitialDataSet {
44
4
        let mut data = InitialDataSet::default();
45
4
        let mut customer_names = HashSet::new();
46
787
        for customer_id in 0..gen_range(rng, self.number_of_customers.clone()) {
47
787
            let customer = Customer::fake(rng, &customer_names);
48
787
            customer_names.insert(customer.name.clone());
49
787
            data.customers.insert(customer_id, customer);
50
787
        }
51
52
        for category_id in 0..gen_range(rng, self.number_of_categories.clone()) {
52
52
            data.categories.insert(category_id, Category::fake(rng));
53
52
        }
54
4
        let mut product_names = HashSet::new();
55
462
        for product_id in 0..gen_range(rng, self.number_of_products.clone()) {
56
462
            let product = Product::fake(rng, &data.categories, &product_names);
57
462
            product_names.insert(product.name.clone());
58
462
            data.products.insert(product_id, product);
59
462
        }
60
4
        let customer_ids = data.customers.keys().copied().collect::<Vec<_>>();
61
4
        let product_ids = data.products.keys().copied().collect::<Vec<_>>();
62
535
        for order_id in 0..gen_range(rng, self.number_of_orders.clone()) {
63
535
            data.orders
64
535
                .insert(order_id, Order::fake(rng, &customer_ids, &product_ids));
65
535
        }
66
4
        let products_available_to_rate = data
67
4
            .orders
68
4
            .values()
69
535
            .map(|order| {
70
535
                order
71
535
                    .product_ids
72
535
                    .iter()
73
4789
                    .map(|&product_id| (order.customer_id, product_id))
74
535
            })
75
4
            .flatten()
76
4
            .collect::<Vec<_>>();
77
4
        let mut rated_products = HashSet::new();
78
4
        if !products_available_to_rate.is_empty() {
79
4
            for _ in 0..gen_range(rng, self.number_of_reviews.clone()) {
80
443
                if let Some(review) =
81
443
                    ProductReview::fake(rng, &products_available_to_rate, &mut rated_products)
82
443
                {
83
443
                    data.reviews.push(review);
84
443
                }
85
            }
86
        }
87

            
88
4
        data
89
4
    }
90
}
91

            
92
#[derive(Serialize, Deserialize, Debug)]
93
struct BenchmarkReport {}
94

            
95
2361
#[derive(Clone, Debug, Serialize, Deserialize)]
96
pub struct Customer {
97
    pub name: String,
98
    pub email: String,
99
    pub address: String,
100
    pub city: String,
101
    pub region: String,
102
    pub country: String,
103
    pub postal_code: String,
104
    pub phone: String,
105
}
106

            
107
impl Customer {
108
787
    pub fn fake<R: Rng>(rng: &mut R, taken_names: &HashSet<String>) -> Self {
109
787
        let name = loop {
110
787
            let name = format!(
111
787
                "{} {}",
112
787
                Name(EN).fake_with_rng::<String, _>(rng),
113
787
                rng.gen::<u8>()
114
787
            );
115
787
            if !taken_names.contains(&name) {
116
787
                break name;
117
            }
118
        };
119
787
        Self {
120
787
            name,
121
787
            email: SafeEmail(EN).fake_with_rng(rng),
122
787
            address: format!(
123
787
                "{} {} {}",
124
787
                BuildingNumber(EN).fake_with_rng::<String, _>(rng),
125
787
                StreetName(EN).fake_with_rng::<String, _>(rng),
126
787
                StreetSuffix(EN).fake_with_rng::<String, _>(rng)
127
787
            ),
128
787
            city: CityName(EN).fake_with_rng(rng),
129
787
            region: StateName(EN).fake_with_rng(rng),
130
787
            country: CountryCode(EN).fake_with_rng(rng),
131
787
            postal_code: PostCode(EN).fake_with_rng(rng),
132
787
            phone: PhoneNumber(EN).fake_with_rng(rng),
133
787
        }
134
787
    }
135
}
136

            
137
66405
#[derive(Clone, Debug, Serialize, Deserialize)]
138
pub struct Product {
139
    pub name: String,
140
    pub category_ids: Vec<u32>,
141
}
142

            
143
impl Product {
144
462
    pub fn fake<R: Rng>(
145
462
        rng: &mut R,
146
462
        available_categories: &BTreeMap<u32, Category>,
147
462
        taken_names: &HashSet<String>,
148
462
    ) -> Self {
149
462
        let mut available_category_ids = available_categories.keys().copied().collect::<Vec<_>>();
150
462
        let number_of_categories = if available_category_ids.is_empty() {
151
            0
152
        } else {
153
462
            rng.gen_range(0..(available_category_ids.len() + 1) / 2)
154
        };
155
462
        let mut category_ids = Vec::with_capacity(number_of_categories);
156
1334
        for _ in 0..number_of_categories {
157
1334
            let category_index = rng.gen_range(0..available_category_ids.len());
158
1334
            category_ids.push(available_category_ids.remove(category_index));
159
1334
        }
160

            
161
462
        let name = loop {
162
462
            let name = format!(
163
462
                "{} {}",
164
462
                Bs(EN).fake_with_rng::<String, _>(rng),
165
462
                rng.gen::<u8>()
166
462
            );
167
462
            if !taken_names.contains(&name) {
168
462
                break name;
169
            }
170
        };
171

            
172
462
        Self { name, category_ids }
173
462
    }
174
}
175

            
176
156
#[derive(Clone, Debug, Serialize, Deserialize)]
177
pub struct Category {
178
    pub name: String,
179
}
180

            
181
impl Category {
182
52
    pub fn fake<R: Rng>(rng: &mut R) -> Self {
183
52
        Self {
184
52
            name: BsAdj(EN).fake_with_rng(rng),
185
52
        }
186
52
    }
187
}
188

            
189
1923
#[derive(Clone, Debug, Serialize, Deserialize)]
190
pub struct Order {
191
    pub customer_id: u32,
192
    pub product_ids: Vec<u32>,
193
}
194

            
195
impl Order {
196
535
    pub fn fake<R: Rng>(
197
535
        rng: &mut R,
198
535
        available_customers: &[u32],
199
535
        available_products: &[u32],
200
535
    ) -> Self {
201
535
        let mut available_product_ids = available_products.to_vec();
202
535
        let number_of_products = if available_product_ids.is_empty() {
203
            0
204
        } else {
205
535
            rng.gen_range(0..((available_product_ids.len() + 2) / 3).min(20))
206
        };
207
535
        let mut product_ids = Vec::with_capacity(number_of_products);
208
4789
        for _ in 0..number_of_products {
209
4789
            let product_index = rng.gen_range(0..available_product_ids.len());
210
4789
            product_ids.push(available_product_ids.remove(product_index));
211
4789
        }
212

            
213
535
        Self {
214
535
            customer_id: *available_customers.choose(rng).unwrap(),
215
535
            product_ids,
216
535
        }
217
535
    }
218
}
219

            
220
17280
#[derive(Clone, Debug, Serialize, Deserialize, Default)]
221
pub struct Cart {
222
    pub customer_id: Option<u32>,
223
    pub product_ids: Vec<u32>,
224
}
225

            
226
14067
#[derive(Clone, Debug, Serialize, Deserialize)]
227
pub struct ProductReview {
228
    pub customer_id: u32,
229
    pub product_id: u32,
230
    pub rating: u8,
231
    pub review: Option<String>,
232
}
233

            
234
impl ProductReview {
235
443
    pub fn fake<R: Rng>(
236
443
        rng: &mut R,
237
443
        available_customer_products: &[(u32, u32)],
238
443
        already_rated: &mut HashSet<(u32, u32)>,
239
443
    ) -> Option<Self> {
240
479
        while available_customer_products.len() > already_rated.len() {
241
479
            let index = rng.gen_range(0..available_customer_products.len());
242
479
            let (customer_id, product_id) = available_customer_products[index];
243
479
            if already_rated.insert((customer_id, product_id)) {
244
                return Some(Self {
245
443
                    customer_id,
246
443
                    product_id,
247
443
                    rating: rng.gen_range(1..=5),
248
443
                    review: if rng.gen_bool(0.25) {
249
95
                        Some(Vec::<String>::dummy_with_rng(&Paragraphs(EN, 1..5), rng).join("\n\n"))
250
                    } else {
251
348
                        None
252
                    },
253
                });
254
36
            }
255
        }
256

            
257
        None
258
443
    }
259
}