This is a Ruby implementation of Stringent. The name Stringent was already taken within RubyGems so this was named Stringento instead.
Stringento
This library provides a pluggable string templating system. At its core, it can take a templated string (such as: 'last_name, first_name middle_name') and a input object (default is a plain old Ruby object) and it will dynamically resolve the string based on the input. It also provides two optional arguments which help make this library 'pluggable':
- Custom Resolver: Instead of passing in a PORO as your input, you can pass in other object types and use a custom resolver for value getting.
- Custom Formatter: Customize the way each string token is rendered by extending the token syntax to: token:formatter:argument. Formatter is the formatting method to call and argument will be passed in as meta-data/options.
Installation
To install through Rubygems:
gem install install stringento
You can also add this to your Gemfile:
bundle add stringento
Examples
Getting Started
Consider the following example:
'The fox_speed brown fox jumps over the dog_speed dog'
We can evaluate this by executing:
example = 'The {fox_speed} brown fox jumps over the {dog_speed} dog'
input = { 'fox_speed' => 'quick', 'dog_speed' => 'lazy' }
result = Stringento.evaluate(example, input)
Result should now be: 'The quick brown fox jumps over the lazy dog'
Custom Resolution
The above example works just fine using the default object value resolver, but you can also pass in a custom function that will serve as resolver. That way you could use dot notation for value resolution. For example:
class NestedHashResolver
def resolve(value, input)
parts = value.to_s.split('.').map(&:to_sym)
input ? input.dig(*parts) : nil
end
end
example = 'The {fox.speed} brown fox jumps over the {dog.speed} dog'
input = { fox: { speed: 'quick' }, dog: { speed: 'lazy' } }
result = Stringento.evaluate(example, input, resolver: NestedHashResolver.new)
The result
variable should now be: 'The quick brown fox jumps over the lazy dog'
Custom Formatting
Another extendable aspect is formatting. Consider a basic example where we want to show 'Yes' for a true value, 'No' for a false value, and an 'Unknown' for a null value:
'The fox is quick: foxfox.quickfox.quick::yes_no_unknown'
In this case we can use a custom formatter called yes_no_unknown that understands how to do this for us. The custom formatter is controlled externally from this library, which means you can use this for whatever purpose you see fit. Here would be an example implementation of this custom formatter:
class CustomFormatter < Stringento::Formatter
def yes_no_unknown_formatter(value, _arg)
if value.nil?
'Unknown'
elsif value
'Yes'
else
'No'
end
end
end
Now, we can pass this in and consume it as follows:
class NestedHashResolver
def resolve(value, input)
parts = value.to_s.split('.').map(&:to_sym)
input ? input.dig(*parts) : nil
end
end
example = 'The fox is quick: {fox.quick::yes_no_unknown}'
input = { fox: { quick: true } }
result = Stringento.evaluate(example, input, resolver: NestedHashResolver.new, formatter: CustomFormatter.new)
The result
variable should now be: 'The fox is quick: Yes'
Note: If we wanted to use the default resolver we could omit resolver instead of the custom resolver.
Contributing
Development Environment Configuration
Basic steps to take to get this repository compiling:
- Install Ruby (check stringento.gemspec for versions supported)
- Install bundler (gem install bundler)
- Clone the repository (git clone [email protected]:bluemarblepayroll/stringento.git)
- Navigate to the root folder (cd stringento)
- Install dependencies (bundle)
Running Tests
To execute the test suite run:
bundle exec rspec spec --format documentation
Alternatively, you can have Guard watch for changes:
bundle exec guard
Also, do not forget to run Rubocop:
bundle exec rubocop
Publishing
Note: ensure you have proper authorization before trying to publish new versions.
After code changes have successfully gone through the Pull Request review process then the following steps should be followed for publishing new versions:
- Merge Pull Request into master
- Update
lib/stringento/version.rb
using semantic versioning - Install dependencies:
bundle
- Update
CHANGELOG.md
with release notes - Commit & push master to remote and ensure CI builds master successfully
- Run
bundle exec rake release
, which will create a git tag for the version, push git commits and tags, and push the.gem
file to rubygems.org.
Code of Conduct
Everyone interacting in this codebase, issue trackers, chat rooms and mailing lists is expected to follow the code of conduct.
License
This project is MIT Licensed.