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
use raii::Raii;
use {ffi, Handle, Return};
use super::types::OdbcType;
pub unsafe trait Output<'a>: Sized {
fn get_data(
stmt: &mut Raii<ffi::Stmt>,
col_or_param_num: u16,
buffer: &'a mut Vec<u8>,
) -> Return<Option<Self>>;
}
unsafe impl<'a, T> Output<'a> for T
where
T: OdbcType<'a>,
{
fn get_data(
stmt: &mut Raii<ffi::Stmt>,
col_or_param_num: u16,
buffer: &'a mut Vec<u8>,
) -> Return<Option<Self>> {
stmt.get_data(col_or_param_num, buffer)
}
}
impl<'p> Raii<'p, ffi::Stmt> {
fn get_data<'a, T>(
&mut self,
col_or_param_num: u16,
buffer: &'a mut Vec<u8>
) -> Return<Option<T>>
where
T: OdbcType<'a>,
{
self.get_partial_data(col_or_param_num, buffer, 0)
}
fn get_partial_data<'a, T>(
&mut self,
col_or_param_num: u16,
buffer: &'a mut Vec<u8>,
start_pos: usize
) -> Return<Option<T>>
where
T: OdbcType<'a>,
{
if buffer.len() - start_pos == 0 {
panic!("buffer length may not be zero");
}
if buffer.len() - start_pos > ffi::SQLLEN::max_value() as usize {
panic!("buffer is larger than {} bytes", ffi::SQLLEN::max_value());
}
let mut indicator: ffi::SQLLEN = 0;
let result = unsafe { ffi::SQLGetData(
self.handle(),
col_or_param_num,
T::c_data_type(),
buffer.as_mut_ptr().offset(start_pos as isize) as ffi::SQLPOINTER,
(buffer.len() - start_pos) as ffi::SQLLEN,
&mut indicator as *mut ffi::SQLLEN,
) };
match result {
ffi::SQL_SUCCESS => {
if indicator == ffi::SQL_NULL_DATA {
Return::Success(None)
} else {
assert!(start_pos + indicator as usize <= buffer.len(), "no more data but indicatior outside of data buffer");
let slice = &buffer[..(start_pos + indicator as usize)];
Return::Success(Some(T::convert(slice)))
}
}
ffi::SQL_SUCCESS_WITH_INFO => {
let initial_len = buffer.len();
let null_offset = T::null_bytes_count();
if indicator == ffi::SQL_NO_TOTAL {
buffer.resize(initial_len * 2, 0);
return self.get_partial_data(col_or_param_num, buffer, initial_len - null_offset);
} else {
if indicator >= initial_len as ffi::SQLLEN {
buffer.resize(indicator as usize + T::null_bytes_count(), 0);
return self.get_partial_data(col_or_param_num, buffer, initial_len - null_offset);
} else {
let slice = &buffer[..(start_pos + indicator as usize)];
Return::SuccessWithInfo(Some(T::convert(slice)))
}
}
}
ffi::SQL_ERROR => Return::Error,
ffi::SQL_NO_DATA => panic!("SQLGetData has already returned the colmun data"),
r => panic!("unexpected return value from SQLGetData: {:?}", r),
}
}
}