- Works in conjunction with the FMDB framework to allow developers to work with first-class objects
- Builds frameworks for iOS and Mac projects
- Generates and updates SQLite database files and tables automatically based on your class definitions
- Can be used with existing SQLite database files
- Install with Swift Package Manager
- Swift 5
- Xcode 10.1 or higher
DBBBuilder is a Swift project that makes it easy to work with first-class objects that are persisted to an SQLite database with the FMDB framework in your iOS or Mac projects.
DBBBuilder takes care of creating and updating the database file, including all tables and indexes if needed, at runtime, based on your class definitions and other configuration steps.
You can use Swift Package Manager to add DBBBuilder to your iOS, WatchOS and macOS projects. To do so, with your project open in Xcode, choose File>Swift Packages>Add Package Dependency... and enter https://github.com/dennisbirch/dbbbuilder-swift
in the text box of the Choose Package Respository dialog that appears. Then hit the Next or Finish button on this and all subsequent screens until you see that the DBBBuilder library has been added to your project.
The workspace in this repository includes demo projects and unit tests for iOS and macOS targets. You can examine the code in these projects to get guidance on using DBBBuilder. You should be able to run the tests from either the iOS or macOS scheme. You can run either the iOS or macOS demo by selecting it from the Scheme menu.
The workspace also includes a macOS target named CodeGenerator that speeds up writing the required boilerplate for initializing one of the primary classes. You can read more about that below.
DBBBuilder has two main classes and some supporting types that you need to be concerned with to use it properly.
DBBManager
is the class that holds a reference to the database file you're persisting data to. You can have more than one DBBManager instance in your project, but each one must be set up to work with a different file.
DBBTableObject
is the parent class for each object you want to persist to the database.
Proper use of DBBBuilder requires setting up each DBBManager instance with the DBBTableObject subclass types it manages. You also need to configure the DBBManager with mapping for each DBBTableObject subclass, defining the properties that should be persisted to the database file and their types. You can optionally configure the DBBManager with indexing instructions for any DBBTableObject subclass to tell it which properties it should build indexes for.
Initalizing: init(databaseURL: URL)
databaseURL: The URL for a database file. Be sure it points to a file and not to a directory.
Configuring with table types:
public func addTableClasses(_ tableClasses: [DBBTableObject.Type])
tableClasses: An array of DBBTableObject types.
After initializing a DBBManager instance, tell it what DBBTableObject subclasses it will include in its database table with this method. For example:
let dbManager = DBBManager(databaseURL: dbFileURL)
// Person, Project and Meeting are DBBTableObject subclasses
let tableClasses = [Person.self, Project.self, Meeting.self]
dbManager.addTableClasses(tableClasses)
Defining subclass properties: func addPersistenceMapping(_ contents: [String : DBBPropertyPersistence], for tableObject: DBBTableObject, indexer: DBBIndexer? = nil)
contents: A [String : DBBPropertyPersistence] dictionary defining the name and type of each property in a DBBTableObject subclass that should be persisted to the database.
tableObject: A DBBTableObject subclass. If you call this method from a DBBSubclass's init method, you can provide self as the parameter.
indexer: An optional argument that takes an instance of a DBBIndexer
struct, with information on building indexes for properties belonging to this subclass. See the discussion of DBBIndexer below.
It is best practice to call this method in your DBBTableObject subclass instances' init methods. See below.
There are a number of public helper functions that are documented in the DBBManager.swift file itself.
Defining properties: Because DBBBuilder uses Key Value Observing to assign values from the database to DBBTableObject subclass instances, each of a DBBTableObject subclass's properties that you want to persist to the database must be marked with the @objc
attribute.
For example:
class Project: DBBTableObject {
@objc var name = ""
@objc var code = ""
@objc var startDate: Date?
@objc var endDate: Date?
@objc var budget: Float = 0
@objc var meetings = [Meeting]()
@objc var tags = [String]()
@objc var subProject: Project?
@objc var projectLead: Person?
Initializing: required init(dbManager: DBBManager)
dbManager: The DBBManager instance managing the database file that this subclass's instances should be written to.
A DBBTableObject subclass must be initialized with this init method.
As part of the initialization process, you should call DBBManager's addPersistenceMapping(_contents: tableObject: indexer:) method to configure the manager with the metadata it needs for writing subclass property values to the database file.
This method takes as its contents parameter, a [String : DBBPropertyPersistence] dictionary. The String key is the case-sensitive name of the property. DBBProperty is a simple struct that defines a database column name and its type as one of a member of the DBBStorageType enum. This allows you to assign a custom name for a property in your class so that the property name and databse column name can differ if necessary. For example if you're working with a pre-existing table where you want the property name to be different from the column name already defined, you can use the DBBPropertyPeristence init method variation that accepts a column name to map the datatabase's column name to your object's property name.
A complete DBBTableObject subclass's init method might look like this:
required init(dbManager: DBBManager) {
super.init(dbManager: dbManager)
let index = DBBIndexer(columnNames: ["name"])
let map: [String : DBBPropertyPersistence] = ["name" : DBBPropertyPersistence(type: .string, columnName: "itemName"),
"code" : DBBPropertyPersistence(type: .string),
"startDate" : DBBPropertyPersistence(type: .date),
"endDate" : DBBPropertyPersistence(type: .date),
"budget" : DBBPropertyPersistence(type: .float),
"meetings" : DBBPropertyPersistence(type: .dbbObjectArray(objectType: Meeting.self)),
"tags" : DBBPropertyPersistence(type: .stringArray),
"subProject" : DBBPropertyPersistence(type: .dbbObject(objectType: Project.self)),
"projectLead" : DBBPropertyPersistence(type: .dbbObject(objectType: Person.self))]
dbManager.addPersistenceMapping(map, for: self, indexer: indexer)
}
You can see more examples of DBBTableObject initialization in the demo projects included in the DBBBuilder workspace.
Helper tool: There is an Xcode project in the workspace named CodeGenerator that can automatically generate much of the boilerplate for a DBBTableObject subclass, given the subclass name and a list of properties. See the README in its folder for usage directions.
Adding indexes: If you want to include an index for any column (i.e. property), create a DBBIndex instance with that property's name in the indexer's columNames String array property, and include that as the optional indexer argument in the addPersistenceMapping(_contents: table: indexer:) method call. You can add indexes for any property in a DBBTableObject subclass. If a property is defined as a DBBTableObject subclass or as an array, DBBBuilder stores data in a join table and automatically indexes that table.
Additional properties: DBBTableObject has two other properties you may want to take advantage of in different situations.
attributesDictionary: [String : String]?
An optional dictionary that lets you define attributes to apply to class properties. The dictionary's keys are property names, and values are attributes to their database columns, either NOT NULL or DISTINCT. See the section on Helpers for more.
isDirty
A Boolean value you can set and read from your application code to use to determine whether an object's state is altered.
DBBPropertyPersistence is used as part of the process of initializing a DBBManager instance to define the properties that should be persisted to the database file. As explained above, it is best practice to add this meta-data as part of a DBBTableObject subclass's initialization process.
initialization: public init(type: DBBStorageType)
type: A member of the DBBStorageType enum that defines the property's type. See below.
Or: public init(type: DBBStorageType, columnName: String)
type: Same as above.
columnName: A String argument that defines an alternate name to use in the database file to define this property. This is useful if you're working with an existing database table and want or need to have a different name for a corresponding property.
DBBIndexer is used to contain the names of all database columns that should be indexed for a table. You can also set the indexer's unique property to true if the index should only work on unique values.
initialization: public init(columnsToIndex: [String], unique: Bool = false)
columnsToIndex: A String array of names of properties that should be indexed in the database file.
unique: An optional Bool value telling DBBBuilder to create UNIQUE indexes. It is set to False by default.
An enum that defines the types DBBBuilder can persist to the database file. You use these types to configure your DBBManager instance(s) with mapping that tells them what to persist to file and how to persist it.
- .bool - Boolean values
- .int - All Integer types
- .float - All floating point numbers
- .string - Strings
- .date - Dates
- .dbbObject - Any DBBTableObject subclass
- .boolArray - Array of Boolean values
- .intArray - Array of Integer types
- .floatArray - Array of floating point numbers
- .stringArray - Array of Strings
- .dateArray - Array of Dates
- .dbbObjectArray - Array of DBBTableObject subclass
The dbbObject and dbbObjectArray members take an objectType
argument in their initializers. This is a specific subclass type.
See the DBBTableObject discussion above for an example of DBBStorageType usage.
There are two ways of saving DBBTableObject subclass instances to the database, one at a time, or as an array of homogenous objects.
public func saveToDB() -> Bool
Saves a single instance to the database. This method returns a Boolean value indicating whether the save succeeded. In case of failure you can get an error message from the underlying database by calling the errorMessage function on your DBManager's instance. For example:
print(dbMgr.errorMessage())
public static func saveObjects(_ objects: [DBBTableObject], dbManager: DBBManager) -> Bool
A static method for saving an array of DBTableObject subclass instances to the database file.
objects: A homogenous array of DBBTableObject subclass types. If you pass in an array of different types, the method will abort before saving anything.
dbManager: The DBBManager instance managing the database the object values should be saved to.
Returns: A Boolean indicating successful execution.
An example of using this method:
let people = [joe, mary, billy]
let success = Person.saveObjects(people, dbManager: mgr)
if success == false {
print("Error saving person array: \(mgr.errorMessage())")
}
There are two additional public methods you can call from your application's DBBTableObject subclasses:
public func performPreSaveActions()
If you override this method, it will be called immediately before data is inserted into the database or the database is updated. You can perform whatever actions are necessary to prepare instances for saving in this method.
public func performPostSaveActions()
If you override this method, it will be called immediately after data is inserted into the database or the database is updated. You can perform whatever actions are necessary to continue after instances are persisted.
There are several methods for retrieving objects from the database. All of them are static methods that should be called on your DBBTableObject subclass types. There are also variations that can be called asynchronously to provide greater safety in cases where you do not control which thread the call might be coming from. Each method returns (or provides as an argument to a completion handler) either a single DBBTableObject instance or an array of them. You will need to cast them to their subclass type in order to use them in most cases. For example:
guard let projects = Project.allInstances(manager: manager) as? [Project] else {
return
}
... or...
Project.getAllInstances(manager: manager: completion { instances in
guard let projects = instances as? [Project] else {
return
}
})
public static func allInstances(manager: DBBManager) -> [DBBTableObject]
A static method to retrieve all instances of a DBBTableObject subclass from the database.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
Returns: An array of all instances of the subclass in the database, fully populated.
public static func instancesWithOptions(_ options: DBBQueryOptions, manager: DBBManager, sparsePopulation: Bool = false) -> [DBBTableObject]?
A static method to retrieve all instances of a DBBTableObject subclass from the database asyncronously.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
completion: A closure with the signature ([DBBTableObject]) -> Void
. When the function completes, the DBBTableObject instances can be obtained at the calling site as the closure's argument.
public static func getAllInstancesFromQueue(manager: DBBManager, completion: ([DBBTableObject]) -> Void)
A static method to retrieve DBBTableObjects that meet defined criteria.
options: A DBBQueryOptions instance that defines characteristics of the DBBTableObjects you want to retrieve. See the section on DBBQueryOptions below for details.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
sparsePopulation: An optional Boolean value indicating whether you would like returned objects to be populated only with their id and created and modified dates. The default value is False.
Returns: An optional array of instances that meet the parameters passed in as options.
Note: You can query the database for objects that match the ID(s) of a property that is of type DBBTableObject. To do so, include a condition in the options argument with the name of the property and the ID number it should match. For example, if you have a Meeting DBBTableObject that has as a 'project' property, which is a Project DBBTableObject subclass, you could include a condition like "project = (projectID)" (where 'projectID' is an Integer value). If the property is represented as an array of DBBTableObjects, you should use "IN" syntax: "project IN ((idsArray))" (where 'idsArray' is an array of Integer values).
See the testMeeting method in the MeetingTests.swift file for examples of these usages.
public static func getInstancesFromQueue(withOptions options: DBBQueryOptions, manager: DBBManager, sparsePopulation: Bool = false, completion: ([DBBTableObject], NSError?) -> Void)
A static method to get an array of DBBTableObjects that meet requested criteria in a completion handler.
options: A DBBQueryOptions instance that defines characteristics of the DBBTableObjects you want to retrieve. See the section on DBBQueryOptions below for details.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
sparsePopulation: An optional Boolean value indicating whether you would like returned objects to be populated only with their id and created and modified dates. The default value is False.
completion: A closure that is called when the fetch is complete. At the calling site you'll get an array of DBBTableObject subclasses that meet your criteria, and an optional error, which if non-nil, provides information on the reason for the failure.
public static func instanceWithIDNumber(_ id: Int64, manager: DBBManager, sparsePopulation: Bool = false) -> DBBTableObject?
A static method to get a single DBBTableObject subclass instance matching the ID number passed in.
id: An Int64 value representing the id number for the instance you want to retrieve.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
sparsePopulation: An optional Boolean value indicating whether you would like the returned object to be populated only with its id and created and modified dates. The default value is False.
Returns: An optional DBBTableObject instance whose id property matches the id value passed in.
public static func getQueuedInstanceWithIDNumber(_ id: Int64, manager: DBBManager, completion: (DBBTableObject?) -> Void, sparsePopulation: Bool = false)
A static method to get an instance of a DBBTableObject sublass by idNum via an asynchronous queue.
id: An Int64 value representing the id number for the instance you want to retrieve.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
completionHandler: A closure with the signature (DBBTableObject?) -> Void
. When the function completes, the DBBTableObject instance can be obtained at the calling site as the closure's argument, unless it fails to find the desired instance in which case the value will be nil.
sparsePopulation: An optional Boolean value indicating whether you would like the returned object to be populated only with its id and created and modified dates. The default value is False.
public static func instancesWithIDNumbers(_ ids: [Int64], manager: DBBManager) -> [DBBTableObject]
A static method to get an array of DBBTableObject subclass instances matching the ID numbers passed in.
ids: An array of Int64 values representing the id numbers for the instances you want to retrieve.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
sparsePopulation: An optional Boolean value indicating whether you would like returned objects to be populated only with their id and created and modified dates. The default value is False.
Returns: An array of DBBTableObject instance whose id properties match the id values passed in.
public static func getInstancesWithIDNumbersFromQueue(ids: [Int64], manager: DBBManager, completion: ([DBBTableObject]) -> Void)
A static method that lets you retrieve an array of instances of a DBTableObject subclass for the idNum
values you pass in via an asynchronous queue.
ids: An array of Int64 values representing the id numbers for the instances you want to retrieve.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
completionHandler: A closure with the signature ([DBBTableObject]) -> Void
. When the function completes, the DBBTableObject instances can be obtained at the calling site as the closure's argument.
public static func allInstanceIDs(manager: DBBManager) -> [Int64]
A static method to retrieve the ids of all instances of a DBBTableObject subclass from the database.
manager: A DBBManager instance that owns the FMDB instance/SQLite file being read from.
Returns: An array of Int64 values representing all the id values for the subclass you ran the request on.
There are multiple DBBTableObject static methods for deleting rows associated with object instances from their database tables. You call them on the DBBTableObject subclass type of the instance(s) you want to delete.
public static func deleteInstance(_ instance: DBBTableObject, manager: DBBManager) -> Bool
Deletes a single instance from the database. This method returns a Boolean value indicating whether the deletion succeeded.
instance: The instance of the DBBTableObject subclass type you want to delete.
dbManager: The DBBManager instance managing the database the object values should be saved to.
Returns: A Boolean indicating successful execution.
In case of failure you can get an error message from the underlying database by calling the errorMessage function on your DBManager's instance. For example:
print(dbMgr.errorMessage())
public static func deleteMultipleInstances(_ instances: [DBBTableObject], manager: DBBManager) -> Bool
A static method for deleting an array of DBTableObject subclass instances from the database file.
instances: A homogenous array of instances of the DBBTableObject subclass type you want to delete.
dbManager: The DBBManager instance managing the database the object values should be saved to.
Returns: A Boolean indicating successful execution.
public static func deleteMultipleInstancesOnQueue(_ instances: [DBBTableObject], manager: DBBManager, completion: (Bool) -> Void)
A static method to delete multiple instances of a DBBTableObject subclass on an asynchronous queue.
instances: Array containing the objects you want to delete. manager: The instance of the DBBManager that owns the FMDatabase instance and SQLite file containing the object to be deleted. completion: A closure with a Bool argument which signals success of the deletion operation
public static func deleteAllInstances(manager: DBBManager) -> Bool
A static method for deleting all instances of a DBBTableObject type.
dbManager: The DBBManager instance managing the database the object values should be saved to.
Returns: A Boolean indicating successful execution.
When retrieving objects from the database, some methods take a DBBQueryOptions argument to determine which and how objects should be returned.
The DBBQueryOptions struct has five properties that can be set to influence fetch results for methods that take a DBBQueryOptions argument.
conditions: An optional array of strings specifying conditions to match on, which are added to a WHERE clause. Each array item should define one condition to match on, e.g. "score > 50". By default, condition clauses are created with AND (i.e. matches all conditions) logic. If you instead want OR logic (i.e. matches any condition), add "OR" as an item in the array.
propertyNames: An optional array of strings for properties to include in the results. Properties that are omitted do not have their values populated. This can enhance performance if the object graph is complex and not all properties are needed in certain cases.
sorting: An optional array of strings for column sort orders, by sort priorty. All columns receive the same ascending or descending order. You can change the default ascending sort order to descending by including the ColumnSorting.descending
value (defined in DBBBuilder.swift) as one of the items in the array.
joinPropertiesToPopulate: An optional array of strings for properties from join tables (arrays, binary values and DBBTableObject types) to include in the results. Properties that are omitted do not have their values populated. This can enhance performance if the object graph is complex and not all properties are needed in certain cases.
distinct: A boolean value for specifying distinct return values. The default for this option is False.
There are several convenience methods for getting partially populated DBBQueryOptions instances. If you declare the instance as a var at the calling site, you can modify other properties as desired.
public static func options(withConditions conditions: [String]? = nil, properties: [String]? = nil, sortColumns: [String]? = nil, ascendingSort: Bool = true, distinct: Bool = false) -> DBBQueryOptions
Convenience method to create a DBBQueryOptions instance with conditions, property name, and sorting arrays, or any combination thereof.
conditions: A string array with the conditions that should be met. Optional.
properties: A string array with the names of properties to include in query results. Optional.
sortColumns: A string array with the names of columns results should be sorted on, in priority order. Optional.
ascendingSort: A Bool value indicating whether columns should be sorted in ascending order. Optional. The default value is True.
distinct: An optional Bool value specifying whether it should be a DISTINCT fetch. Optional. The default value is False.
Returns: A DBBQueryOptions instance with the specified options set to the input received.
public static func queryOptionsWithPropertyNames(_ properties: [String], distinct: Bool = false) -> DBBQueryOptions
Method to get a DBBQueryOptions instance with the propertyNames values passed in, and optionally its distinct value.
properties: A string array of property names to populate.
distinct: An optional Boolean value to specify returning only distinct values. Pass True if you want to override the default False value.
public static func queryOptionsWithAscendingSortForColumns(_ columnNames: [String]) -> DBBQueryOptions
Method to get a DBBQueryOptions instance for sorting in ascending order by the columns passed in, and optionally its distinct value.
columnNames: A string array of column names to determine sorting order.
distinct: An optional Boolean value to specify returning only distinct values. Pass True if you want to override the default False value.
public static func queryOptionsWithDescendingSortForColumns(_ columnNames: [String]) -> DBBQueryOptions
Method to get a DBBQueryOptions instance for sorting in descending order by the columns passed in, and optionally its distinct value.
columnNames: A string array of column names to determine sorting order.
distinct: An optional Boolean value to specify returning only distinct values. Pass True if you want to override the default False value.
public static func queryOptionsWithConditions(_ conditions: [String], distinct: Bool = false) -> DBBQueryOptions
Method to get a DBBQueryOptions instance with conditions to match on, and optionally its distinct value.
conditions: A string array of conditions to match on.
distinct: An optional Boolean value to specify returning only distinct values. Pass True if you want to override the default False value.
NOTE: When using a condition clause, you should sanitize any strings you pass in as part of a condition. Because DBBBuilder creates the WHERE clause dynamically, it cannot automatically escape strings for you. There is a publicly accessible extension on String in the framework that you can use for this purpose, public func dbb_SQLEscaped() -> String
. Example usage:
let isBossCondition = "\(person.employer) = \(company.owner.dbb_SQLEscaped())"
let options = DBBQueryOptions.queryOptionsWithConditions([isBossCondition])
If you are writing several objects of the same type at once, you can optimize persisting them to the database by using the DBBTableObject static method saveObjects(objects:dbManager), instead of persisting them one at a time.
There are a couple of strategies you can pursue to enhance the performance of object retrieval. If your object graph is complex, you may want to consider one of these strategies to improve the user experience.
Sparse object population: Some of the object retrieval methods have an optional sparsePopulation argument which defaults to False. If you pass in True for this value, you will get back objects which only have their id, createdTime, and modifiedTime values populated. This would allow you to then fetch objects with only required properties populated (see Explicit object population below), based on their ID number.
Explicit object population: Some of the object retrieval methods let you specify what properties should be populated with a DBBQueryOptions instance. This is a less severe form of sparse population that can still improve performance for complex objects. You could benefit from this approach when you need to fetch many objects, but only need to display basic information to the user in a view.
The DBBManager class offers a couple of methods you can use to determine and set a version for the SQLite database managed by the DBBManager to help with database migration. While DBBBuilder automatically checks and updates tables to conform to their corresponding DBBTableObject specifications, it does not delete existing columns or populate new columns with data. If a schema is being updated sufficiently that you need to perform such changes when you update object definitions, you'll need to perform these steps manually. Using the methods below, you might want to add a version check routine that follows a pattern such as:
- After database setup is complete, check the current database version
- If it equals a defined value, exit your checking routine
- If it needs to be updated with any table population or column deletion, call out to an update method that performs those tasks
- In the update method, set the current database version to the newly defined value
public func hasLatestDBVersion(currentVersion: Double) -> (hasLatest: Bool, version: Double)
A helper method for checking a version number (which you must assign) for determining whether a database migration is required.
currentVersion: A Double value representing the version you want to check against.
Returns: A (Bool, Double) tuple with values for whether or not the current database version is equal to or greater than the currentVersion value passed in, and the version the database is set to.
public func setCurrentDBVersion(_ version: Double)
A helper method for setting a version number for the database, which can be useful for determining the need for database migrations in conjunction with the hasLatestDB
function. If the database already has a version number equal to or higher than the value passed in, it does NOT set the value passed in.
currentVersion: A Double value to set the database version to.
There are some helper methods and definitions available in different classes that may be useful at various points of development.
ColumnAttributes: There is a publicly accessible ColumnAttributes struct in the DBBBuilder.swift file that defines notNull = "NOT NULL"
and unique = "UNIQUE"
which you may want to use if setting a DBBTableObject subclass's attributesDictionary property.
ColumnSorting: There is a publicly accessible ColumnSorting struct in the DBBBuilder.swift file that defines ascending = "ASC
and descending = "DESC"
which you may want to use for assigning sort order query options.
Error messages: public func errorMessage() -> String
A publicly accessible method in DBBManager for getting the error message from the underlying SQLite database for any operation on it.
Table/object count: public func countForTable(_ tableName: String) -> Int
A publicly accessible method in DBBManager for getting the count of rows in any table.
tableName: The name of the DBBTableObject subclass whose count you want to access.
Returns: The number of rows (object instances) in the database file for that subclass.
Table name: There is a publicly available computed property on DBBTableObject that returns the subclass's short name, which is used as the table name in the database that DBBBuilder creates.
Database cleanup: public func vacuumDB()
You may want to periodically clean up the database file. You can do so by calling this helper method in the DBBManager class.
Dropping indexes: public func dropIndex(named indexName: String) -> Bool
If you ever want to delete an index from a database file, you can call this method to do so.
indexName: The name of the index to drop.
Date Extensions
public func dbb_dateComparisonString() -> String
When called on a Date instance, this method returns a string you can use in conditional clauses comparing dates.
Example usage:
let lastOccurrence = Date()addingTimeInterval(-lastOccurrenceTimeInterval)
let conditions = ["\(entryDate.dbb_DateComparisonString()) > \(lastOccurrence.dbb_DateComparisonString())"]
let options = DBBQueryOptions.queryOptionsWithConditions([conditions])
if let recentEvents = Event.instancesWithOptions(options, manager: manager) as? [Event] {
// continue with additional steps
}
func dbb_dateFromTimeInterval(_ interval: TimeInterval)
For the greatest accuracy, DBBBuilder stores dates in its SQLite files as TimeIntervals (Doubles). You probably should not need to use this method to directly translate an underlying value to a Date, but it's available if required.
static func dbb_dateFromTimeInterval(_ interval: TimeInterval) -> Date
This static function on Date provides the inverse functionality, returning a Date instance from the TimeInterval passed in as its single argument.
String Extensions
public func dbb_SQLEscaped() -> String
When called on a string, returns a version escaped for use in a SQLite query. For example, "John Brown's body".dbb_SQLEscaped() would return 'John Brown''s body'.