Marten 0.4.0 release notes
January 13, 2024.
Requirements and compatibility
Crystal 1.9, 1.10, and 1.11.
New features
Generators
Marten now provides a generator mechanism that makes it easy to create various abstractions, files, and structures within an existing project. This feature is available through the use of the gen management command and facilitates the generation of key components such as models, schemas, emails, or applications. The authentication application can now also be added easily to existing projects through the use of generators. By leveraging generators, developers can improve their workflow and speed up the development of their Marten projects while following best practices.
Below are highlighted some examples illustrating the use of the gen management command:
# Generate a model in the admin app:
marten gen model User name:string email:string --app admin
# Generate a new TestEmail email in the blog application:
marten gen email TestEmail --app blog
# Add a new 'blogging' application to the current project:
marten gen app blogging
# Add the authentication application to the current project:
margen gen auth
You can read more about the generator mechanism in the dedicated documentation. All the available generators are also listed in the generators reference.
Multi table inheritance
It is now possible to define models that inherit from other concrete models (ie. non-abstract models). In this situation, each model can be used/queried individually and has its own associated database table. The framework automatically defines a set of "links" between each model that uses multi table inheritance and its parent models in order to ensure that the relational structure and inheritance hierarchy are maintained.
For example:
class Person < Marten::Model
field :id, :big_int, primary_key: true, auto: true
field :first_name, :string, max_size: 100
field :last_name, :string, max_size: 100
end
class Employee < Person
field :company_name, :string, max_size: 100
end
employee = Employee.filter(first_name: "John").first!
employee.first_name # => "John"
All the fields defined in the Person model can be accessed when interacting with records of the Employee model (despite the fact that the data itself is stored in distinct tables).
You can read more about this new kind of model inheritance in Multi table inheritance.
Schema handler callbacks
Handlers that inherit from the base schema handler - Marten::Handlers::Schema - or one of its subclasses (such as Marten::Handlers::RecordCreate or Marten::Handlers::RecordUpdate) can now define new kinds of callbacks that allow to easily manipulate the considered schema instance and to define logic to execute before the schema is validated or after (eg. when the schema validation is successful or failed):
before_schema_validationafter_schema_validationafter_successful_schema_validationafter_failed_schema_validation
For example, the after_successful_schema_validation callback can be used to create a flash message after a schema has been successfully validated:
class ArticleCreateHandler < Marten::Handlers::Schema
success_route_name "home"
template_name "articles/create.html"
schema ArticleSchema
after_successful_schema_validation :generate_success_flash_message
private def generate_success_flash_message : Nil
flash[:notice] = "Article successfully created!"
end
end
Please head over to Schema handler callbacks to learn more about these new types of callbacks.
URL field for models and schemas
It is now possible to define url fields in models and schemas. These allow you to easily persist valid URLs in your models but also to expect valid URL values in data validated through the use of schemas.
For example:
class User < Marten::Model
field :id, :big_int, primary_key: true, auto: true
field :website_url, :url, blank: true, null: true
end
Slug field for models and schemas
It is now possible to define slug fields in models and schemas. These allow you to easily persist valid slug values (ie. strings that can only include characters, numbers, dashes, and underscores) in your models but also to expect such values in data validated through the use of schemas.
For example:
class User < Marten::Model
field :id, :big_int, primary_key: true, auto: true
field :username, :slug
end
Minor features
Models and databases
- Support for removing records from many-to-many fields was added and many-to-many field query sets now provide a
#removehelper method allowing to easily remove specific records from a specific relation. You can learn more about this capability in Many-to-many relationships. - Support for clearing all the references to records targeted by many-to-many fields was added. Indeed, many-to-many field query sets now provide a
#clearmethod allowing to easily clear a specific relation. You can learn more about this capability in Many-to-many relationships. - It is now possible to specify arrays of records to add or remove from a many-to-many relationship query set, through the use of the
#addand#removemethods. See the related documentation to learn more about interacting with records targeted by many-to-many relationships. - Records targeted by reverse relations that are contributed to models by
one_to_one(ie. when using therelatedoption) are now memoized when the corresponding methods are called on related model instances. - Relation fields that contribute methods that return query sets to models (such as
many_to_oneormany_to_manyfields) now make sure that those query set objects are memoized at the record level. The corresponding instance variables are also reset when the considered records are reloaded. This allows to limit the number of queries involved when iterating multiple times over the records targeted by amany_to_manyfield for example. - The
#orderquery set method can now be called directly on model classes to allow retrieving all the records of the considered model in a specific order. - A
#pk?model method can now be leveraged to determine if a primary key value is set on a given model record. - The
#joinquery set method now makes it possible to pre-select one-to-one reverse relations. This essentially allows to traverse aone_to_onefield back to the model record on which the field is specified. - The
#countquery set method can now take an optional field name to count the number of records that have a non-null value for the corresponding column in the database.
Handlers and HTTP
- It is now optional to define a name for included route maps (but defining a name for individual routes that are associated with handlers is still mandatory). You can read more about this in Defining included routes.
- The
Marten::Handlers::Schemageneric handler now allows modifying the schema object context name through the use of the#schema_context_namemethod. - It is now possible to specify symbol status codes when making use of the
#respond,#render,#head, and#jsonresponse helper methods. Such symbols must comply with the values of theHTTP::Statusenum. - The hash of matched routing parameters that is available from handlers through the use of the
#paramsmethod can accept symbols and strings when performing key lookups. - The GZip middleware now incorporates a mitigation strategy against the BREACH attack. This strategy (described in the Heal The Breach paper) involves introducing up to 100 random bytes into GZip responses to enhance the security against such attacks.
- A new
before_rendercallback type is now available to handlers. Such callbacks are executed before rendering a template in order to produce a response. As such they are well suited for adding new variables to the global template context so that they are available to the template runtime. - All handlers now have access to a global template context through the use of the
#contextmethod. This template context object is available for the lifetime of the considered handler and can be mutated to define which variables are made available to the template runtime when rendering templates (either through the use of the#renderhelper method or when rendering templates as part of subclasses of theMarten::Handlers::Templategeneric handler). This feature can be combined with thebefore_rendercallback to effortlessly introduce new variables to the context used for rendering a template and generating a handler response.
Templates
- A
withtemplate tag was introduced in order to make it easy to assign one or more variables inside a template block. - A
timetemplate tag was introduced in order to make it possible to output the string representation of a time variable according to a specific time format pattern. - An
escapetemplate tag was introduced in order to make it easy to explicitly escape safe strings in templates. - The ability to configure how undefined/unknown variables are treated was added to the framework: by default, such variables are treated as
nilvalues (so nothing is displayed for such variables, and they are evaluated as falsey in if conditions). This behavior can be configured via thetemplates.strict_variablessetting, and you can learn more about it in Strict variables.
Development
- The
newmanagement command now accepts an optional--databaseoption that can be used to preconfigure the application database (eg.--database=postgresql). - A
clearsessionsmanagement command was introduced in order to ease the process of clearing expired session entries. - Custom management commands can now define how they want to handle unknown or undefined arguments through the use of the
#on_unknown_argumentmethod. This can be leveraged to implement management commands which can accept a variable number of positional arguments. - Custom management commands can now define how they want to handle invalid options through the use of the
#on_invalid_optionmethod.
Emailing
- Emails now provide a set of callbacks that make it easy to define logic that is triggered at different stages of an email's lifecycle (before/after an email gets delivered, before rendering the email's template).
Authentication
- The generated authentication application now features the ability to change the password of the currently logged-in user.
Backward incompatible changes
Handlers and HTTP
- Custom session stores must now implement a
#clear_expired_entriesmethod (allowing to clear expired session entries if this is applicable for the considered store). - The introduction of the global template context involves that generic handlers that used to override the
#contextmethod (in order to insert record or schema objects into the template context for example) now leveragebefore_rendercallbacks in order to mutate the global context and define the same variables. Generic handler subclasses that were overriding this#contextmethod and callingsuperin it will likely need to be updated in order to leverage thebefore_rendercallback to add custom variables to the global template context.
Templates
- The
defaulttemplate filter will now return the specified default value if the incoming value is falsey or empty.
Emailing
- The introduction of the global template context involves that emails that used to explicitly define a
#contextmethod (eg. in order to define a hash of template variables from local instance variables) won't work anymore. Instead these emails should now leverage thebefore_rendercallback in order to add these variables to the email template context object.