surrealpatch/lib/src/dbs/options.rs

227 lines
5 KiB
Rust
Raw Normal View History

use crate::cnf;
use crate::dbs::Auth;
use crate::dbs::Level;
use crate::err::Error;
use std::sync::Arc;
// An Options is passed around when processing a set of query
// statements. An Options contains specific information for how
// to process each particular statement, including the record
// version to retrieve, whether futures should be processed, and
// whether field/event/table queries should be processed (useful
// when importing data, where these queries might fail).
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct Options {
// Currently selected NS
pub ns: Option<Arc<str>>,
// Currently selected DB
pub db: Option<Arc<str>>,
// Connection authentication data
pub auth: Arc<Auth>,
// Approximately how large is the current call stack?
dive: u8,
2022-06-27 16:01:39 +00:00
// Whether live queries are allowed?
pub live: bool,
// Should we force tables/events to re-run?
pub force: bool,
// Should we run permissions checks?
pub perms: bool,
// Should we error if tables don't exist?
pub strict: bool,
// Should we process field queries?
pub fields: bool,
// Should we process event queries?
pub events: bool,
// Should we process table queries?
pub tables: bool,
// Should we process index queries?
pub indexes: bool,
// Should we process function futures?
pub futures: bool,
}
impl Default for Options {
fn default() -> Self {
Options::new(Auth::No)
}
}
impl Options {
2022-09-16 08:55:18 +00:00
/// Create a new Options object
pub fn new(auth: Auth) -> Options {
Options {
ns: None,
db: None,
dive: 0,
2022-06-27 16:01:39 +00:00
live: false,
perms: true,
force: false,
strict: false,
fields: true,
events: true,
tables: true,
indexes: true,
futures: false,
auth: Arc::new(auth),
}
}
2022-09-16 08:55:18 +00:00
/// Get currently selected NS
2022-02-26 00:35:11 +00:00
pub fn ns(&self) -> &str {
self.ns.as_ref().unwrap()
}
2022-09-16 08:55:18 +00:00
/// Get currently selected DB
2022-02-26 00:35:11 +00:00
pub fn db(&self) -> &str {
self.db.as_ref().unwrap()
}
/// Create a new Options object for a function/subquery/future/etc.
///
/// The parameter is the approximate cost of the operation (more concretely, the size of the
/// stack frame it uses relative to a simple function call). When in doubt, use a value of 1.
pub fn dive(&self, cost: u8) -> Result<Options, Error> {
let dive = self.dive.saturating_add(cost);
if dive <= cnf::MAX_COMPUTATION_DEPTH {
Ok(Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
dive,
..*self
})
} else {
Err(Error::ComputationDepthExceeded)
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn force(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
force: v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn perms(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
perms: v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn fields(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
fields: v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn events(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
events: v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn tables(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
tables: v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn indexes(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
indexes: v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn import(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
fields: !v,
events: !v,
tables: !v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn strict(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
strict: v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Create a new Options object for a subquery
pub fn futures(&self, v: bool) -> Options {
Options {
auth: self.auth.clone(),
ns: self.ns.clone(),
db: self.db.clone(),
futures: v,
..*self
}
}
2022-09-16 08:55:18 +00:00
/// Check whether realtime queries are supported
2022-06-27 16:01:39 +00:00
pub fn realtime(&self) -> Result<(), Error> {
if !self.live {
return Err(Error::RealtimeDisabled);
}
Ok(())
}
2022-09-16 08:55:18 +00:00
/// Check whether the authentication permissions are ok
pub fn check(&self, level: Level) -> Result<(), Error> {
if !self.auth.check(level) {
return Err(Error::QueryPermissions);
}
Ok(())
}
2022-09-16 08:55:18 +00:00
/// Check whether the necessary NS / DB options have been set
pub fn needs(&self, level: Level) -> Result<(), Error> {
if self.ns.is_none() && matches!(level, Level::Ns | Level::Db) {
return Err(Error::NsEmpty);
}
if self.db.is_none() && matches!(level, Level::Db) {
return Err(Error::DbEmpty);
}
Ok(())
}
}