-
Notifications
You must be signed in to change notification settings - Fork 73
/
Copy pathextdedup.rs
309 lines (272 loc) · 10.8 KB
/
extdedup.rs
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
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
static USAGE: &str = r#"
Remove duplicate rows from an arbitrarily large CSV/text file using a memory-mapped,
on-disk hash table.
Unlike the 'dedup' command, this command does not load the entire file into memory
to sort the CSV first before deduping it.
This allows it to run in constant memory and the output will retain the input sort order.
This command has TWO modes of operation.
* CSV MODE
when --select is set, it dedupes based on the given column/s. See `qsv select --help`
for select syntax details.
* LINE MODE
when --select is NOT set, it deduplicates any input text file (not just CSVs) on a
line-by-line basis.
A duplicate count will be sent to <stderr>.
Usage:
qsv extdedup [options] [<input>] [<output>]
qsv extdedup --help
extdedup options:
-s, --select <arg> Select a subset of columns to dedup.
Note that the outputs will remain at the full width of the CSV.
If --select is NOT set, extdedup will work in LINE MODE, deduping
the input as a text file on a line-by-line basis.
--no-output Do not write deduplicated output to <output>.
Use this if you only want to know the duplicate count.
-D, --dupes-output <file> Write duplicates to <file>.
Note that the file will NOT be a valid CSV.
It is a list of duplicate lines, with the row number of the
duplicate separated by a tab from the duplicate line itself.
-H, --human-readable Comma separate duplicate count.
--memory-limit <arg> The maximum amount of memory to buffer the on-disk hash table.
If less than 50, this is a percentage of total memory.
If more than 50, this is the memory in MB to allocate, capped
at 90 percent of total memory.
[default: 10]
Common options:
CSV MODE ONLY:
-n, --no-headers When set, the first row will not be interpreted
as headers. That is, it will be deduped with the rest
of the rows. Otherwise, the first row will always
appear as the header row in the output.
-d, --delimiter <arg> The field delimiter for reading CSV data.
Must be a single character. (default: ,)
-h, --help Display this message
-q, --quiet Do not print duplicate count to stderr.
"#;
use std::{
fs,
io::{self, stdin, stdout, BufRead, Write},
};
use indicatif::HumanCount;
use serde::Deserialize;
use sysinfo::System;
use crate::{
config,
config::{Config, Delimiter},
odhtcache,
select::SelectColumns,
util, CliResult,
};
#[derive(Deserialize)]
struct Args {
arg_input: Option<String>,
flag_select: Option<SelectColumns>,
arg_output: Option<String>,
flag_no_headers: bool,
flag_delimiter: Option<Delimiter>,
flag_no_output: bool,
flag_dupes_output: Option<String>,
flag_human_readable: bool,
flag_memory_limit: Option<u64>,
flag_quiet: bool,
}
const MEMORY_LIMITED_BUFFER: u64 = 100 * 1_000_000; // 100 MB
pub fn run(argv: &[&str]) -> CliResult<()> {
let args: Args = util::get_args(USAGE, argv)?;
// Set the memory buffer size for the on-disk hash table based on --memory-limit
// and system capabilities.
let mem_limited_buffer_bytes = calculate_memory_limit(args.flag_memory_limit);
log::info!("{mem_limited_buffer_bytes} bytes used for memory buffer for on-disk hash table...");
let quiet = args.flag_quiet;
let human_readable = args.flag_human_readable;
let dupes_count = if args.flag_select.is_some() {
dedup_csv(args, mem_limited_buffer_bytes)?
} else {
dedup_lines(args, mem_limited_buffer_bytes)?
};
if quiet {
return Ok(());
}
eprintln!(
"{}",
if human_readable {
HumanCount(dupes_count).to_string()
} else {
dupes_count.to_string()
}
);
Ok(())
}
fn dedup_csv(args: Args, mem_limited_buffer: u64) -> Result<u64, crate::clitypes::CliError> {
let rconfig = Config::new(args.arg_input.as_ref())
.delimiter(args.flag_delimiter)
.no_headers(args.flag_no_headers)
.select(args.flag_select.unwrap());
let mut rdr = rconfig.reader()?;
let mut wtr = Config::new(args.arg_output.as_ref()).writer()?;
let dupes_output = args.flag_dupes_output.is_some();
let mut dupewtr = Config::new(args.flag_dupes_output.as_ref()).writer()?;
let headers = rdr.byte_headers()?.clone();
if dupes_output {
let mut dupe_headers = csv::ByteRecord::new();
dupe_headers.push_field(b"dupe_rowno");
dupe_headers.extend(headers.iter());
dupewtr.write_byte_record(&dupe_headers)?;
}
let mut dedup_cache = odhtcache::ExtDedupCache::new(mem_limited_buffer);
let mut dupes_count = 0_u64;
let sel = rconfig.selection(&headers)?;
rconfig.write_headers(&mut rdr, &mut wtr)?;
// Pre-allocate and reuse buffers
let mut key = String::with_capacity(20);
let mut utf8_string = String::with_capacity(20);
let mut dupe_row = csv::ByteRecord::new();
let mut curr_row = csv::ByteRecord::new();
for (row_idx, row) in rdr.byte_records().enumerate() {
curr_row.clone_from(&row?);
key.clear();
for field in sel.select(&curr_row) {
if let Ok(s_utf8) = simdutf8::basic::from_utf8(field) {
key.push_str(s_utf8);
} else {
utf8_string.clear();
utf8_string.push_str(&String::from_utf8_lossy(field));
key.push_str(&utf8_string);
}
}
if dedup_cache.contains(&key) {
dupes_count += 1;
if dupes_output {
dupe_row.clear();
dupe_row.push_field(itoa::Buffer::new().format(row_idx + 1).as_bytes());
dupe_row.extend(curr_row.iter());
dupewtr.write_byte_record(&dupe_row)?;
}
} else {
dedup_cache.insert(&key);
wtr.write_byte_record(&curr_row)?;
}
}
dupewtr.flush()?;
wtr.flush()?;
Ok(dupes_count)
}
fn dedup_lines(args: Args, mem_limited_buffer: u64) -> Result<u64, crate::clitypes::CliError> {
let input_reader: Box<dyn BufRead> = match &args.arg_input {
Some(input_path) => {
if input_path.to_lowercase().ends_with(".sz") {
return fail_clierror!(
"Input file cannot be a .sz file. Use 'qsv snappy decompress' first."
);
}
let file = fs::File::open(input_path)?;
Box::new(io::BufReader::with_capacity(
config::DEFAULT_RDR_BUFFER_CAPACITY,
file,
))
},
None => Box::new(io::BufReader::new(stdin().lock())),
};
let mut output_writer: Box<dyn Write> = match &args.arg_output {
Some(output_path) => Box::new(io::BufWriter::with_capacity(
config::DEFAULT_WTR_BUFFER_CAPACITY,
fs::File::create(output_path)?,
)),
None => Box::new(io::BufWriter::with_capacity(
config::DEFAULT_WTR_BUFFER_CAPACITY,
stdout().lock(),
)),
};
let mut write_dupes = false;
#[cfg(target_family = "unix")]
let mut dupes_writer = if let Some(dupes_output) = args.flag_dupes_output {
write_dupes = true;
io::BufWriter::with_capacity(
config::DEFAULT_WTR_BUFFER_CAPACITY,
fs::File::create(dupes_output)?,
)
} else {
io::BufWriter::with_capacity(
config::DEFAULT_WTR_BUFFER_CAPACITY,
fs::File::create("/dev/null")?,
)
};
#[cfg(target_family = "windows")]
let mut dupes_writer = if let Some(dupes_output) = args.flag_dupes_output {
write_dupes = true;
io::BufWriter::with_capacity(
config::DEFAULT_WTR_BUFFER_CAPACITY,
fs::File::create(dupes_output)?,
)
} else {
io::BufWriter::with_capacity(
config::DEFAULT_WTR_BUFFER_CAPACITY,
fs::File::create("nul")?,
)
};
let mut dedup_cache = odhtcache::ExtDedupCache::new(mem_limited_buffer);
let mut dupes_count = 0_u64;
let mut line_work = String::with_capacity(1024);
for (row_idx, line) in input_reader.lines().enumerate() {
line_work.clone_from(&line?);
if dedup_cache.contains(&line_work) {
dupes_count += 1;
if write_dupes {
writeln!(dupes_writer, "{row_idx}\t{line_work}")?;
}
} else {
dedup_cache.insert(&line_work);
if args.flag_no_output {
continue;
}
writeln!(output_writer, "{line_work}")?;
}
}
dupes_writer.flush()?;
output_writer.flush()?;
Ok(dupes_count)
}
/// Determines the memory buffer size to use for on-disk hash table based on
/// the provided flag and the system's total memory.
///
/// # Arguments
///
/// * `flag_memory_limit` - An optional u64 value representing the user-specified memory limit.
///
/// # Returns
///
/// A u64 value representing the calculated memory limit in bytes.
///
/// # Behavior
///
/// - If the system is not supported, it returns a predefined `MEMORY_LIMITED_BUFFER` value.
/// - If `flag_memory_limit` is None, it returns the `MEMORY_LIMITED_BUFFER`.
/// - If `flag_memory_limit` is Some(limit):
/// - For limit <= 50, it's treated as a percentage of total system memory.
/// - For limit > 50, it's treated as megabytes, but capped at 90% of total system memory.
pub fn calculate_memory_limit(flag_memory_limit: Option<u64>) -> u64 {
if !sysinfo::IS_SUPPORTED_SYSTEM {
return MEMORY_LIMITED_BUFFER;
}
let mut sys = System::new();
sys.refresh_memory();
let total_memory = sys.total_memory();
#[allow(clippy::cast_precision_loss)]
match flag_memory_limit {
Some(limit) if limit <= 50 => ((total_memory as f64 * limit as f64) / 100.0) as u64,
Some(limit) => {
let limit_bytes = limit.saturating_mul(1_000_000); // Convert MB to bytes
let ninety_percent_total = (total_memory as f64 * 0.9) as u64;
std::cmp::min(limit_bytes, ninety_percent_total)
},
None => MEMORY_LIMITED_BUFFER,
}
}
#[test]
fn test_extdedup_mem_check() {
// check to see if sysinfo return meminfo without segfaulting
let mut sys = System::new();
sys.refresh_memory();
let mem10percent = (sys.total_memory() * 1000) / 10; // 10 percent of total memory
assert!(mem10percent > 0);
}