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
use crate::{handles::Statement, Error, InputParameter, Parameter}; mod tuple; /// SQL Parameters used to execute a query. /// /// ODBC allows to place question marks (`?`) in the statement text as placeholders. For each such /// placeholder a parameter needs to be bound to the statement before executing it. /// /// # Examples /// /// This trait is implemented by single parameters. /// /// ```no_run /// use odbc_api::Environment; /// /// let env = unsafe { /// Environment::new()? /// }; /// /// let mut conn = env.connect("YourDatabase", "SA", "<YourStrong@Passw0rd>")?; /// let year = 1980; /// if let Some(cursor) = conn.execute("SELECT year, name FROM Birthdays WHERE year > ?;", &year)? { /// // Use cursor to process query results. /// } /// # Ok::<(), odbc_api::Error>(()) /// ``` /// /// Tuples of `Parameter`s implement this trait, too. /// /// ```no_run /// use odbc_api::Environment; /// /// let env = unsafe { /// Environment::new()? /// }; /// /// let mut conn = env.connect("YourDatabase", "SA", "<YourStrong@Passw0rd>")?; /// let too_old = 1980; /// let too_young = 2000; /// if let Some(cursor) = conn.execute( /// "SELECT year, name FROM Birthdays WHERE ? < year < ?;", /// (&too_old, &too_young), /// )? { /// // Use cursor to congratulate only persons in the right age group... /// } /// # Ok::<(), odbc_api::Error>(()) /// ``` /// /// And so do array slices of `Parameter`s. /// /// ```no_run /// use odbc_api::Environment; /// /// let env = unsafe { /// Environment::new()? /// }; /// /// let mut conn = env.connect("YourDatabase", "SA", "<YourStrong@Passw0rd>")?; /// let params = [1980, 2000]; /// if let Some(cursor) = conn.execute( /// "SELECT year, name FROM Birthdays WHERE ? < year < ?;", /// ¶ms[..])? /// { /// // Use cursor to process query results. /// } /// # Ok::<(), odbc_api::Error>(()) /// ``` pub unsafe trait ParameterCollection { /// Number of values per parameter in the collection. This can be different from the maximum /// batch size a buffer may be able to hold. Returning `0` will cause the the query not to be /// executed. fn parameter_set_size(&self) -> u32; /// # Safety /// /// Implementers should take care that the values bound by this method to the statement live at /// least for the Duration of `self`. The most straight forward way of achieving this is of /// course, to bind members. unsafe fn bind_parameters_to(self, stmt: &mut Statement) -> Result<(), Error>; } unsafe impl<T> ParameterCollection for T where T: Parameter, { fn parameter_set_size(&self) -> u32 { 1 } unsafe fn bind_parameters_to(self, stmt: &mut Statement) -> Result<(), Error> { self.bind_parameter(1, stmt) } } unsafe impl<T> ParameterCollection for &[T] where T: InputParameter, { fn parameter_set_size(&self) -> u32 { 1 } unsafe fn bind_parameters_to(self, stmt: &mut Statement) -> Result<(), Error> { for (index, parameter) in self.iter().enumerate() { stmt.bind_input_parameter(index as u16 + 1, parameter)?; } Ok(()) } }