This is an ORM similar to ActiveRecord, but uses the filesystem instead.
It uses TMSU which is a filesystem tagging system.
Usage:
Install the gem
gem install tmsu_file_db
Define a model
require 'tmsu_file_db'
class User < TmsuModel
# this configure block is optional
# it defaults to a randomly named dir in ./db
configure root_path: "./db/users"
# Validations must return an array
validate do |record|
record.name.nil? ? ["name can't be blank"] : []
end
# Specific attributes can be validated as well
validate(:email) do |email, record|
email&.include?("@") ? ["email isn't valid"] : []
end
end
Create instances
# create, update, and delete
u = User.new name: "max"
u.valid? # => false
u.errors # => ["email isn't valid"]
u.save # => false
u.[:email] = "[email protected]"
u.valid? # => true
u.save # => true
u.update(email: "[email protected]") # => true
u.update(email: "") # => false
# There are getter methods for convenience
# Setters need to use []=
u.name # => "max"
u["name"] # => "max"
u[:name] # => "max"
u[:name] # => "max p."
# All these getter/setters are working on 'attributes' under the hood.
u.attributes[:name] # => "max p."
# each record is assigned a filesystem path
u.path
# creating a new record will create a new file in the root_path
# but will not add anything to the file unless .write is called
u.write "hello"
# The content of the file is not part of the "attributes" i.e. name and email
# Those are stored using TMSU tags
u.tags # => { email: "[email protected]", name: "max" }
u.tags == u.attributes # true
# Attributes can be deleted
u.delete :name
u.tags # => { email: "[email protected]" }
# Records can be deleted (this will destroy the file)
u.destroy
Use class-level query methods
Note that this does not use Arel or any of that jazz. So chaining queries or using joins will not work.
Note also that there is no id
on models, only path
, which is an absolute path.
User.where(name: "max p.")[0].name == "max p." # => true
User.find_by(name: "max p.").name == "max p." # => true
User.update_all(name: "max") # => true
# You can make arbitrary queries using TMSU syntax
# e.g. select all users with email set that are not named melvin
User.query("name != 'melvin' and email")[0].name == "max" # => true
Although there's an index method (all
), there's no typical auto-incrementing ids stored in TMSU. So to load a single, arbitrary record without tags, its file path is used:
u.path # => "./db/users/23uj8d9j328dj"
User.from_file(u.path).name == "max p."
Use TmsuRuby.file
An alternative to TmsuModel
is to use TmsuRuby.file
instead. This does not handle creation / deletion of files. It should only be used with files that already exist.
Note that these methods are technically available on TmsuModel
instances as well. But this shouldn't be done, because it will cause the in-memory attributes to be out of sync. Also, some operations like tag
will error if called on unsaved records.
file_path = './my_pic.jpg' # this should already exist
tmsu_file = TmsuRuby.file file_path
tmsu_file.tags # => {}
tmsu_file.tag "foo" # .tag can be passed a string
tmsu_file.tags # => { foo: nil }
tmsu_file.untag "foo"
tmsu_file.tags # => { }
tmsu_file.tag ["foo", "bar"] # .tag can also be passed an array
tmsu_file.tags # => { foo: nil, bar: nil }
tmsu_file.tag(a: 1, b: 2) # .tag can also be passed a hash
tmsu_file.tags # => { foo: nil, bar: nil, a: 1, b: 2 }
It's also possible to use TmsuRuby
to work on multiple files instead of just one:
glob_selector = "./**/*.jpg"
tmsu_file = TmsuRuby.file glob_selector
# there is a special method used to add tags in this case
tmsu_file.tag_selector "foo"
tmsu_file.tag_selector ["a", "b"]
tmsu_file.tag_selector c: 1, d: 2
# Simiarly to untag
tmsu_file.untag_selector "c"
# check that the tags were added to files
TmsuRuby.file("./my_pic.jpg").tags
# => { foo: nil, a: nil, b: nil, d: 2 }
Using TmsuRuby.file
you can search by tag as well. All these methods return
an array of absolute paths
query_glob = "./**/*.jpg"
# To perform a scoped search (the same used by .where, .find_by, and .query):
# This is a simple query, but the whole TMSU syntax is available
TmsuRuby.file(query_glob).paths_query("foo")
# Search the whole filesystem for files with tag
TmsuRuby.file.files("foo")
Test && Examples
See automated_tests.rb, which can double as usage examples.