-
Notifications
You must be signed in to change notification settings - Fork 257
Upgrading to Blacklight 8
Blacklight 8 includes some backwards-incompatible changes. Follow this guide to upgrade a Blacklight 7 application to Blacklight 8.
-
Ensure your local application has a passing test suite. If needed, add automated tests for any important functionality, to make it obvious when something is broken.
-
Upgrade to the latest 7.x release so you can see any recent deprecation warnings.
-
Take note of any deprecation warnings originating from Blacklight in the logs and make the suggested changes.
To test a plugin and validate it Run the plugin with the latest version of blacklight/main.
-
@document_list
is no longer provided by Blacklight, becauseBlacklight::SearchService#search_results
now returns a single value rather than a tuple: https://github.com/projectblacklight/blacklight/blob/2430033de0f8c54ef7407e28228b3702f1fd3e0a/app/services/blacklight/search_service.rb#L24 This instance variable was previously aActiveSupport::Deprecation::DeprecatedObjectProxy
-
Blacklight::SearchService#fetch
now returns a single value rather than two values. - Remove
component: true
configuration fromadd_facet_field
inCatalogController
- If you previously used a partials configuration in
CatalogController
like this:config.show.partials = [:index_header, :thumbnail, :index]
and some of the partials were provided by Blacklight, this may no longer work as the partials have been replaced by theDocumentComponent
(https://github.com/projectblacklight/blacklight/blob/main/app/components/blacklight/document_component.rb) You can replace theDocumentComponent
by settingconfig.show.document_component = MyDocumentComponent
. You could also copy the partials from the release-7.x branch into your local application. - If you override
app/views/catalog/_show_main_content.html.erb
, please note that it has changed significantly. Notably, it now renders theDocumentComponent
and no longer callsrender_document_partials
- If you override
app/views/catalog/facet.html.erb
, please note that it has changed significantly. Notably, it now renders theBlacklight::FacetComponent
and no longer callsrender_facet_limit
-
app/views/catalog/_constraints_element.html.erb
is no longer present. Overriding it has no effect. - Explicitly pass the documents to the render_document_index method:
- Deprecation warning: https://github.com/projectblacklight/blacklight/pull/2530/files
Accumulate backwards breaking changes here so they can be turned into upgrade path steps in the document above
- "constraints" is what Blacklight calls the area above search results that tells you all the queries/limits/filters at play in your search, and lets you remove them individually. Sometimes called a kind of "breadcrumbs".
- Plugins often want to add elements to this "constraints" area.
- They have traditionally done this by over-riding helper methods either
render_constraints
orrender_constraints_filters
to callsuper
and concatenate their own additional content. - Multiple different plugins can and have done this simultaneously without stepping on each others toes, each adding something to
super
. - This will no longer work in Blacklight 8, those methods are not available for over-riding.
- Additionally, when a plugin wants to supply it's own "constraint", it has often traditionaly called the
render_constraint_element
helper method, to say "render this constraint in whatever the application style is, even if it's been customized". That method is also no longer available.
- They have traditionally done this by over-riding helper methods either
- Examples:
- Upgrade Path:
- In BL 8 all the constraints are rendered by app/view/catalog/_constraints.html.erb That renders the following components:
-
Blacklight::ConstraintsComponent
this used to berender_constraints
-
Blacklight::ConstraintLayoutComponent
this used to berender_constraints_element
- We could override the view partial to render a new subclass of
Blacklight::ConstraintsComponent
- Q: How? I think it requires overriding a chain "up to the top" at present
-
- Another possibility is to move the
ConstraintsComponent
class to a configuration variable and have the partial use that variable.- I am intrigued by this. But there's still a challenge when multiple different plug-ins all want to add elements to constraints -- which is our actually existing use-case. How do they avoid fighting for control over this component, stepping on each other's toes?
blacklight_range_limit
,geoblacklight
, andblacklight_maps
all want to add an element to constraints.
- I am intrigued by this. But there's still a challenge when multiple different plug-ins all want to add elements to constraints -- which is our actually existing use-case. How do they avoid fighting for control over this component, stepping on each other's toes?
- In BL 8 all the constraints are rendered by app/view/catalog/_constraints.html.erb That renders the following components:
- Unlike "adding element to constraints" above, only one thing at a time can do this, there's only one "query" constraint. But plugins have sometimes done this, and so have local apps.
- By over-riding the
render_constraints_query
helper method -- this is no longer available. - Examples:
- Upgrade Path:
- In BL 8 the constraints query is rendered by the query_constraint_component argument (default
Blacklight::ConstraintLayoutComponent
), which is passed into the Blacklight::Constrains component: https://github.com/projectblacklight/blacklight/blob/fd33c74a378d164a571174fa0054c25b70ef09d1/app/components/blacklight/constraints_component.rb#L11 ; To customize, provide a different component.- Q: The challenge for me is "how". I think it may require over-riding a chain of parent components which we aren't actually interested in customizing, to get to the point where a different component can be provided here.
- In BL 8 the constraints query is rendered by the query_constraint_component argument (default
- A plugin or app may want to render search constraints on it's own new page.
- Previously, it could call the helper method
render_constraints
to render the constraints including any customizations added by plugins or local app, as above. - This is no longer available as a method.
- Examples:
- Upgrade Path:
- In BL 8. Render
Blacklight::ConstraintsComponent
.-
Q: The challenge for me is how this interacts with above scenarios -- right now, a plugin wants to render search constraints 'however it may have been customized by the app and it's plugins', and can do so by calling the helper method (which may have been customized with overrides). But in the new scenario, if a plugin renders
Blacklight::ConstraintsComponent
-- this does not get customizations made by above suggestions to "provide a different component" forBlacklight::ConstraintsComponent
-
Q: The challenge for me is how this interacts with above scenarios -- right now, a plugin wants to render search constraints 'however it may have been customized by the app and it's plugins', and can do so by calling the helper method (which may have been customized with overrides). But in the new scenario, if a plugin renders
- In BL 8. Render
After trying to wrap my head around the scenarios for a while, and let it role around in there... what if we provide some additional configuration variables, specifically targeted at these constraints-customizing use-cases? [I think this essentially a means of what they call "dependency inversion", via the blacklight configuration object].
There is a little step into the precedent of providing component classes in Blacklight configuration already exists in the BL 8 codebase.
For instance, a document_component
configuration value (with default) is set here and here, and then referenced here and here.
This way, an application (or possibly plugin) can set a document_component
to a new component, via configuration, and have this new component be used by Blacklight... without the need to try to customize every single place that renders a "document_component" (which may require customizing the things that render the things that render a "document_component", etc).
If we keep going with the idea of components in configuration to allow the specific use cases identified to be achieved in a precisely targetted way, we might have:
-
config.query_constraint_component =
Blacklight::ConstraintLayoutComponent
: Can be set to customize the query constraint as blacklight_advanced_search and some local applications do -
config.additional_constraints_components =
[]
: A default empty array, that you can add 0 or more component classes to, to add additional constraints. This way multiple different plugins can add something to this array, and have their thing rendered on the end of constraints area, without having to fight for control of theConstraintsComponent
or interfere with each other. -
Now a plugin that wants to render constraints can just render
Blacklight::ConstraintsComponents
, and still get all the customizations -
But how do all these things actually render the little boxes? Before they could call the
render_constraint_element
helper, to render a constraint element in a standard way -- that if a local app wanted to change that standard way by overriding that helper, the plugins would use the new changed standard way. Now... they can render the newBlacklight::ConstraintLayoutComponent
, but there's no way for an app to change what that does. Do we need aconfig.constraint_element_component
too, or is that getting too crazy?
This plan does have some complexity downsides. And I guess the exact initialization arguments of each of these configurable components would have to become part of Blacklight API. As I guess is already the case for the document_component
configuration.
But this is the only reasonable way I've been able to come up with to support these use cases no worse than previous Blacklight. It is eg a very popular use case for plugins to want to add an element to the "constraints" area, and you might have multiple plugins all wanting to add their own element.
- Add
config.bootstrap_version = 4
toapp/controllers/catalog_controller.rb
in theconfigure_blacklight do |config|
block