decidim 0.21.0 → 0.23.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of decidim might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/README.md +2 -50
- data/docs/advanced/authorship.md +1 -2
- data/docs/advanced/components.md +40 -0
- data/docs/advanced/content_blocks.md +5 -3
- data/docs/advanced/data-picker.md +37 -9
- data/docs/advanced/embeddable.md +2 -1
- data/docs/advanced/endorsable.md +113 -0
- data/docs/advanced/fixing_locales.md +88 -0
- data/docs/advanced/how_to_fix_metrics.md +34 -21
- data/docs/advanced/machine_translation_service.md +12 -0
- data/docs/advanced/metrics.md +7 -9
- data/docs/advanced/newsletter_templates.md +64 -0
- data/docs/advanced/notifications.md +114 -0
- data/docs/advanced/permissions.md +23 -0
- data/docs/advanced/profiling.md +43 -0
- data/docs/advanced/releases.md +114 -0
- data/docs/advanced/share_tokens.md +53 -0
- data/docs/advanced/templates.md +56 -0
- data/docs/advanced/testing.md +3 -0
- data/docs/customization/machine_translations.md +30 -0
- data/docs/customization/maps.md +610 -0
- data/docs/customization/views.md +31 -2
- data/docs/development_guide.md +36 -46
- data/docs/getting_started.md +6 -3
- data/docs/manual-installation.md +4 -0
- data/docs/services/elections_bulletin_board.md +38 -0
- data/docs/services/etherpad.md +1 -1
- data/docs/services/maps.md +362 -0
- data/docs/services/social_providers.md +33 -7
- data/lib/decidim/version.rb +1 -1
- metadata +61 -48
- data/docs/services/geocoding.md +0 -34
@@ -0,0 +1,12 @@
|
|
1
|
+
# Create your own machine translation service
|
2
|
+
|
3
|
+
You can use the `Decidim::Dev::DummyTranslator` service as a base. Any new translator service will need to implement the same API as this class.
|
4
|
+
|
5
|
+
## Integrating with async services
|
6
|
+
|
7
|
+
Some translation services are async, which means that some extra work is needed. This is the main overview:
|
8
|
+
|
9
|
+
- The Translation service will only send the translation request. It should have a way to send what resource, field and target locale are related to that translation.
|
10
|
+
- You'll need to create a custom controller in your application to receive the callback from the translation service when the translation is finished
|
11
|
+
- From that new endpoint, find a way to find the related resource, field and target locale. Then start a `Decidim::MachineTranslationSaveJob` with that data. This job will handle how to save the data in the DB.
|
12
|
+
|
data/docs/advanced/metrics.md
CHANGED
@@ -37,6 +37,7 @@ Metrics calculations must be executed everyday. Some `rake task` have been added
|
|
37
37
|
```ruby
|
38
38
|
bundle exec rake decidim:metrics:list
|
39
39
|
```
|
40
|
+
|
40
41
|
Currently, available metrics are:
|
41
42
|
|
42
43
|
- **users**, created `Users`
|
@@ -78,20 +79,17 @@ persist metrics from all times and types.
|
|
78
79
|
The `decidim_metrics` table has the following fields:
|
79
80
|
|
80
81
|
- `day`: the day for which the metric has been computed.
|
81
|
-
- `metric_type`: the type of the metric. One of: users, proposals,
|
82
|
-
accepted_proposals, supports, assemblies.
|
82
|
+
- `metric_type`: the type of the metric. One of: users, proposals, accepted_proposals, supports, assemblies.
|
83
83
|
- `cumulative`: quantity accumulated to day ”day”.
|
84
84
|
- `quantity`: quantity for the current day, ”day”.
|
85
|
-
- `decidim_organization_id`: the FK to the organization to which this Metric
|
86
|
-
belongs to.
|
87
|
-
- `
|
88
|
-
participatory space to which this Metric belongs to, if any.
|
89
|
-
- `related_object_type` + `related_object_id`: the FK to the object to which
|
90
|
-
this Metric belongs to, if any.
|
85
|
+
- `decidim_organization_id`: the FK to the organization to which this Metric belongs to.
|
86
|
+
- `participatory_space_type` + `participatory_space_id`: the FK to the participatory space to which this Metric belongs to, if any.
|
87
|
+
- `related_object_type` + `related_object_id`: the FK to the object to which this Metric belongs to, if any.
|
91
88
|
- `decidim_category_id`: the FK to the category for this Metric, if any.
|
92
89
|
|
93
90
|
Relations around `decidim_metrics` table:
|
94
|
-
|
91
|
+
|
92
|
+
```ascii
|
95
93
|
+------------------------+
|
96
94
|
+--------------+ | ParticipatoryProcesses |
|
97
95
|
| Organization | +----+------------------------+
|
@@ -0,0 +1,64 @@
|
|
1
|
+
# Newsletter templates
|
2
|
+
|
3
|
+
The newsletter templates allow the user to select a template amongst a set of of them, and use it as base to send newsletters. This allow for more customization on newsletter, tematic newsletters, etc.
|
4
|
+
|
5
|
+
Code-wise, they use the content blocks system internally, so [check the docs](https://github.com/decidim/decidim/blob/master/docs/advanced/content_blocks.md) for that section first.
|
6
|
+
|
7
|
+
## Adding a new template
|
8
|
+
|
9
|
+
You'll first need to register the template as a content block, but specifying `:newsletter_template` as its scope:
|
10
|
+
|
11
|
+
```ruby
|
12
|
+
Decidim.content_blocks.register(:newsletter_template, :my_template) do |content_block|
|
13
|
+
content_block.cell "decidim/newsletter_templates/my_template"
|
14
|
+
content_block.settings_form_cell = "decidim/newsletter_templates/my_template_settings_form"
|
15
|
+
content_block.public_name_key "decidim.newsletter_templates.my_template.name"
|
16
|
+
|
17
|
+
content_block.images = [
|
18
|
+
{
|
19
|
+
name: :main_image,
|
20
|
+
uploader: "Decidim::NewsletterTemplateImageUploader",
|
21
|
+
preview: -> { ActionController::Base.helpers.asset_path("decidim/placeholder.jpg") }
|
22
|
+
}
|
23
|
+
]
|
24
|
+
|
25
|
+
content_block.settings do |settings|
|
26
|
+
settings.attribute(
|
27
|
+
:body,
|
28
|
+
type: :text,
|
29
|
+
translated: true,
|
30
|
+
preview: -> { ([I18n.t("decidim.newsletter_templates.my_template.body_preview")] * 100).join(" ") }
|
31
|
+
)
|
32
|
+
end
|
33
|
+
end
|
34
|
+
```
|
35
|
+
|
36
|
+
You'll need to add this into an initializer. Note that if you're adding this from a module, then you need to add it from the `engine.rb` file (check the docs for content blocks for more info).
|
37
|
+
|
38
|
+
This is the simplest template. It has a single attribute, in this case a translatable chunk of text. Let's go line by line.
|
39
|
+
|
40
|
+
Inside the block, first we define the path of the cell we'll use to render the email publicly. This cell will receive the `Decidim::ContentBlock` object, which will contain the attributes and any image we have (one in this example). In order to render cells, please note that emails have a very picky HTML syntax, so we suggest using some specialized tools to design the template, export the layout to HTML and render that through the cell. We suggest you make this cell inherit from `Decidim::NewsletterTemplates::BaseCell` for convenience.
|
41
|
+
|
42
|
+
Then we define the cell that will be used to render the form in the admin section. This form needs to show inputs for the attributes defined when registering the template. It will receive the `form` object to render the input fields. We suggest this cell to inherit from `Decidim::NewsletterTemplates::BaseSettingsFormCell`.
|
43
|
+
|
44
|
+
In the third line inside the block we define the I18n path to the public name of the template. This name will serve as identifier for the users who write the newsletters, so be sure to make it descriptive.
|
45
|
+
|
46
|
+
After that we define the images this newsletter supports. We give it a unique name, the class name of the uploader we'll use (the example one is the default one, but you might want to customize this value) and a way to preview this image. This preview image will only be used in the "Preview" page of the template, it's not a fallback. If you want a fallback, please implement it through a custom uploader (see `carrierwave`'s docs for that). There's no limit of the amount of images you add.
|
47
|
+
|
48
|
+
Finally, we define the attributes for the newsletter. In this case we define a body attribute, which is a translatable text. Whether this text require an editor or not will be defined by the settings cell. We also have a way to preview that attribute. There's no limit on the number of attributes you define.
|
49
|
+
|
50
|
+
## Interpolating the recipient name
|
51
|
+
|
52
|
+
Decidim accepts `%{name}` as a placeholder for the recipient name. If you want your template to use it, you'll need to call `parse_interpolations` in your public cell:
|
53
|
+
|
54
|
+
```ruby
|
55
|
+
class Decidim::NewsletterTemplates::MyTemplate < Decidim::ViewModel
|
56
|
+
include Decidim::NewslettersHelper
|
57
|
+
|
58
|
+
def body
|
59
|
+
parse_interpolations(uninterpolated_body, recipient_user, newsletter.id)
|
60
|
+
end
|
61
|
+
end
|
62
|
+
```
|
63
|
+
|
64
|
+
The newsletter subject is automatically interpolated.
|
@@ -0,0 +1,114 @@
|
|
1
|
+
# Notifications
|
2
|
+
|
3
|
+
In Decidim, notifications may mean two things:
|
4
|
+
|
5
|
+
- he concept of notifying an event to a user. This is the wider use of "notification".
|
6
|
+
- the notification's participant space, which lists the `Decidim::Notification`s she has received.
|
7
|
+
|
8
|
+
So, in the wider sense, notifications are messages that are sent to the users, admins or participants, when something interesting occurs in the platform.
|
9
|
+
|
10
|
+
Each notification is sent via two communication channels: email and internal notifications.
|
11
|
+
|
12
|
+
## A Decidim Event
|
13
|
+
|
14
|
+
Technically, a Decidim event is nothing but an `ActiveSupport::Notification` with a payload of the form
|
15
|
+
|
16
|
+
```ruby
|
17
|
+
ActiveSupport::Notifications.publish(
|
18
|
+
event,
|
19
|
+
event_class: event_class.name,
|
20
|
+
resource: resource,
|
21
|
+
affected_users: affected_users.uniq.compact,
|
22
|
+
followers: followers.uniq.compact,
|
23
|
+
extra: extra
|
24
|
+
)
|
25
|
+
```
|
26
|
+
|
27
|
+
To publish an event to send a notification, Decidim's `EventManager` should be used:
|
28
|
+
|
29
|
+
```ruby
|
30
|
+
# Note the convention between the `event` key, and the `event_class` that will be used later to wrap the payload and be used as the email or notification model.
|
31
|
+
data = {
|
32
|
+
event: "decidim.events.comments.comment_created",
|
33
|
+
event_class: Decidim::Comments::CommentCreatedEvent,
|
34
|
+
resource: comment.root_commentable,
|
35
|
+
extra: {
|
36
|
+
comment_id: comment.id
|
37
|
+
},
|
38
|
+
affected_users: [user1, user2],
|
39
|
+
followers: [user3, user4]
|
40
|
+
}
|
41
|
+
|
42
|
+
Decidim::EventsManager.publish(data)
|
43
|
+
```
|
44
|
+
|
45
|
+
Both, `EmailNotificationGenerator` and `NotificationGenerator` are use the same arguments:
|
46
|
+
|
47
|
+
- **event**: A String with the name of the event.
|
48
|
+
- **event_class**: A class that wraps the event.
|
49
|
+
- **resource**: an instance of a class implementing the `Decidim::Resource` concern.
|
50
|
+
- **followers**: a collection of Users that receive the notification because they're following it.
|
51
|
+
- **affected_users**: a collection of Users that receive the notification because they're affected by it.
|
52
|
+
- **force_send**: boolean indicating if EventPublisherJob should skip the `notifiable?` check it performs before notifying.
|
53
|
+
- **extra**: a Hash with extra information to be included in the notification.
|
54
|
+
|
55
|
+
Again, both generators will check for each user
|
56
|
+
|
57
|
+
- in the *followers* array, if she has the `notification_types` set to "all" or "followed-only".
|
58
|
+
- in the *affected_users* array, if she has the `notification_types` set to "all" or "own-only".
|
59
|
+
|
60
|
+
Event names must start with "decidim.events." (the `event` data key). This way `Decidim::EventPublisherJob` will automatically process them. Otherwise no artifact in Decidim will process them, and will be the developer's responsibility to subscribe to them and process.
|
61
|
+
|
62
|
+
Sometimes, when something that must be notified to users happen, a service is defined to manage the logic involved to decide which events should be published. See for example `Decidim::Comments::NewCommentNotificationCreator`.
|
63
|
+
|
64
|
+
Please refer to [Ruby on Rails Notifications documentation](https://api.rubyonrails.org/classes/ActiveSupport/Notifications.html) if you need to hack the Decidim's events system.
|
65
|
+
|
66
|
+
## How Decidim's `EventPublisherJob` processes the events?
|
67
|
+
|
68
|
+
The `EventPublisherJob` in Decidim's core engine subscribes to all notifications matching the regular expression `/^decidim\.events\./`. This is, starting with "decidim.events.". It will then be invoked when an imaginary event named "decidim.events.harmonica_blues" is published.
|
69
|
+
|
70
|
+
When invoked it simply performs some validations and enqueue an `EmailNotificationGeneratorJob` and a `NotificationGeneratorJob`.
|
71
|
+
|
72
|
+
The validations it performs check if the resource, the component, or the participatory space are published (when the concept applies to the artifact).
|
73
|
+
|
74
|
+
## The \*Event class
|
75
|
+
|
76
|
+
Generates the email and notification messages from the information related with the notification.
|
77
|
+
|
78
|
+
Event classes are subclasses of `Decidim::Events::SimpleEvent`.
|
79
|
+
A subset of the payload of the notification is passed to the event class's constructor:
|
80
|
+
|
81
|
+
- The `resource`
|
82
|
+
- The `event` name
|
83
|
+
- The notified user, either from the `followers` or from the `affected_users` arrays
|
84
|
+
- The `extra` hash, with content specific for the given SimpleEvent subclass
|
85
|
+
- The user_role, either :follower or :affected_user
|
86
|
+
|
87
|
+
With the previous information the event class is able to generate the following contents.
|
88
|
+
|
89
|
+
Developers will be able to customize those messages by adding translations to the `config/locales/en.yml` file of the corresponding module.
|
90
|
+
The keys to be used will have the translation scope corresponding to the event name ("decidim.events.comments.comment_by_followed_user" for example) and the key will be the content to override (email_subject, email_intro, etc.)
|
91
|
+
|
92
|
+
### Email contents
|
93
|
+
|
94
|
+
The following are the parts of the notification email:
|
95
|
+
|
96
|
+
- *email_subject*, to be customized
|
97
|
+
- email_greeting, with a good default, usually there's no need to cusomize it
|
98
|
+
- *email_intro*, to be customized
|
99
|
+
- *resource_text* (optional), rendered `html_safe` if present
|
100
|
+
- *resource_url*, a link to the involved resource if resource_url and resource_title are present
|
101
|
+
- *email_outro*
|
102
|
+
|
103
|
+
All contents except the `email_greeting` use to require customization on each notification.
|
104
|
+
|
105
|
+
### Notification contents
|
106
|
+
|
107
|
+
Only the `notification_title` is generated in the event class. The rest of the contents are produced by the templates from the `resource` and the `notification` objects.
|
108
|
+
|
109
|
+
## Testing notifications
|
110
|
+
|
111
|
+
- Test that the event has been published (usually a command test)
|
112
|
+
- Test the event returns the expected contents for the email and the notification.
|
113
|
+
|
114
|
+
Developers should we aware when adding URLs in the email's content, be sure to use absolute URLs and not relative paths.
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# Permissions
|
2
|
+
|
3
|
+
Since Decidim has multiple roles, we needed a permissions system to discover what actions can a user perform, given their roles. The basis of the current permissions system were added on [\#3029](https://github.com/decidim/decidim/pull/3029), so be sure to check that PR (and the related ones) to read the discussion and the motivations behind the change.
|
4
|
+
|
5
|
+
## Overview
|
6
|
+
|
7
|
+
When checking for permission to perform an action, we check this chain:
|
8
|
+
|
9
|
+
1. The component permissions
|
10
|
+
1. The participatory space permissions
|
11
|
+
1. The core permissions
|
12
|
+
|
13
|
+
This way we're going from more specific to more general.
|
14
|
+
|
15
|
+
## Explanation
|
16
|
+
|
17
|
+
We wrap the permission and its context in a `PermissionsAction` object. It also holds the state of the permission (whether it's been allowed or not).
|
18
|
+
|
19
|
+
Each component and space must define a `Permissions` class, inheriting from `Decidim::DefaultPermissions`. The `Permissions` class must define a `permissions` instance method. this class will receive the permission action, and the `permissions` method must return the permission action. The `Permissions` class can set the action as allowed or disallowed.
|
20
|
+
|
21
|
+
There's a small limitation in the permission action state machine: once it's been disallowed it can't be reallowed. This is to avoid mischievous modules modifying permissions.
|
22
|
+
|
23
|
+
Permission actions have a scope. It's usually either `:public` or `:admin`, and the `Permissions` class usually handles the `:public` scope, while it delegates the `:admin` one to another specialized class.
|
@@ -0,0 +1,43 @@
|
|
1
|
+
# How to profile a Decidim app
|
2
|
+
|
3
|
+
The developmen_app includes a bunch of gems that profile the application. Run the following command in the decidim root's folder:
|
4
|
+
|
5
|
+
```bash
|
6
|
+
bundle exec rake development_app
|
7
|
+
```
|
8
|
+
|
9
|
+
and then move into it and boot the server
|
10
|
+
|
11
|
+
```bash
|
12
|
+
cd developmen_app
|
13
|
+
bundle exec rails s
|
14
|
+
```
|
15
|
+
|
16
|
+
## Bullet
|
17
|
+
|
18
|
+
Bullet detects N+1 queries and suggests how to fix them, although it doesn't catch them all. It's currently configured in `config/initializers/bullet.rb` to log in the regular rails log and also in its own `log/bullet.log`. You'll now see entries like the following:
|
19
|
+
|
20
|
+
```bash
|
21
|
+
user: xxx
|
22
|
+
GET /
|
23
|
+
USE eager loading detected
|
24
|
+
Decidim::Comments::Comment => [:author]
|
25
|
+
Add to your query: .includes([:author])
|
26
|
+
Call stack
|
27
|
+
```
|
28
|
+
|
29
|
+
It also warns you when there's an unnecessary eager load.
|
30
|
+
|
31
|
+
More details: https://github.com/flyerhzm/bullet
|
32
|
+
|
33
|
+
## Rack-mini-profiler
|
34
|
+
|
35
|
+
This gem can analyze memory, database, and call stack with flamegraphs. It will show up in development on the top left corner and it gives you all sorts of profiling insights about that page. It'll tell you where the response time was spend on in the call stack.
|
36
|
+
|
37
|
+
This gem is further enhanced with the `flamegraph`, `stackprof` and `memory_profiler` gems which provide more detailed analysis. Try out by appending `?pp=flamegraph`, `?pp=profile-gc` or `?pp=analyze-memory` to the URL. You can read more about these options at https://github.com/MiniProfiler/rack-mini-profiler#flamegraphs.
|
38
|
+
|
39
|
+
More details: https://github.com/MiniProfiler/rack-mini-profiler
|
40
|
+
|
41
|
+
## Profiling best practices
|
42
|
+
|
43
|
+
You need to take the insights of these gems with a grain of salt though, if you run this in development. Rails' development settings have nothing to do with a production set up where classes are not reloaded and assets are precompiled and served from a web server. Therefore, you should mimic these settings as much as possible if you want your findings to be realistic.
|
@@ -0,0 +1,114 @@
|
|
1
|
+
# Releasing new versions
|
2
|
+
|
3
|
+
In order to release new version you need to be owner of all the gems at RubyGems, ask one of the owners to add you before releasing. (Try `gem owner decidim`).
|
4
|
+
|
5
|
+
Remember to follow the Gitflow branching workflow.
|
6
|
+
|
7
|
+
## Create the stable branch for the release
|
8
|
+
|
9
|
+
1. Go to develop with `git checkout develop`
|
10
|
+
1. Create the release branch `git checkout -b release/x.y.z-stable && git push origin release/x.y.z-stable`.
|
11
|
+
1. If required, add the release branch to Crowdin so that any pending translations will generate a PR to this branch.
|
12
|
+
|
13
|
+
Mark `develop` as the reference to the next release:
|
14
|
+
|
15
|
+
1. Go back to develop with `git checkout develop`
|
16
|
+
1. Turn develop into the `dev` branch for the next release:
|
17
|
+
1. Update `.decidim-version` to the new `dev` development version: `x.y.z.dev`, where `x.y.z` is the new semver number for the next version
|
18
|
+
1. Run `bin/rake update_versions`, this will update all references to the new version.
|
19
|
+
1. Run `bin/rake bundle`, this will update all the `Gemfile.lock` files
|
20
|
+
1. Run `bin/rake webpack`, this will update the JavaScript bundle.
|
21
|
+
1. Update `SECURITY.md` and change the supported version to the new version.
|
22
|
+
1. Update the `CHANGELOG.MD`. At the top you should have an `Unreleased` header with the `Added`, `Changed`, `Fixed` and `Removed` empty sections. After that, the header with the current version and link with the same beforementioned sections and a `Previous versions` header at the bottom that links to the previous minor release stable branch.
|
23
|
+
|
24
|
+
```markdown
|
25
|
+
## [Unreleased](https://github.com/decidim/decidim/tree/HEAD)
|
26
|
+
|
27
|
+
### Added
|
28
|
+
|
29
|
+
### Changed
|
30
|
+
|
31
|
+
### Fixed
|
32
|
+
|
33
|
+
### Removed
|
34
|
+
|
35
|
+
## [v0.XX.0](https://github.com/decidim/decidim/releases/tag/v0.XX.0)
|
36
|
+
|
37
|
+
### Added
|
38
|
+
...
|
39
|
+
|
40
|
+
## Previous versions
|
41
|
+
|
42
|
+
Please check [0.XX-stable](https://github.com/decidim/decidim/blob/release/0.XX-stable/CHANGELOG.md) for previous changes.
|
43
|
+
```
|
44
|
+
|
45
|
+
1. Push the changes `git add . && git commit -m "Bump develop to next release version" && git push origin develop`
|
46
|
+
|
47
|
+
## Release Candidates
|
48
|
+
|
49
|
+
Release Candidates are the same as beta versions. They should be ready to go to production, but publicly released just before in order to be widely tested.
|
50
|
+
|
51
|
+
If this is a **Release Candidate version** release, the steps to follow are:
|
52
|
+
|
53
|
+
1. Checkout the release stable branch `git checkout release/x.y-stable`.
|
54
|
+
1. Update `.decidim-version` to the new version `x.y.z.rc1`
|
55
|
+
1. Run `bin/rake update_versions`, this will update all references to the new version.
|
56
|
+
1. Run `bin/rake bundle`, this will update all the `Gemfile.lock` files
|
57
|
+
1. Run `bin/rake webpack`, this will update the JavaScript bundle.
|
58
|
+
1. Commit all the changes: `git add . && git commit -m "Bump to rcXX version" && git push origin release/x.y-stable`.
|
59
|
+
1. Tag the release candidate with `git tag vX.Y.Z.rcN && git push origin vX.Y.Z.rcN`.
|
60
|
+
1. Usually, at this point, the release branch is deployed to meta-decidim during, at least, one week to validate the stability of the version.
|
61
|
+
|
62
|
+
### During the validation period
|
63
|
+
|
64
|
+
1. During the validation period, bugfixes must be implemented directly to the current `release/x.y.z-stable` branch and ported to `develop`.
|
65
|
+
1. During the validation period, translations to the officially supported languages must be added to Crowdin and, when completed, merged into `release/x.y.z-stable`.
|
66
|
+
|
67
|
+
## Major/Minor versions
|
68
|
+
|
69
|
+
Release Candidates will be tested in a production server (usually meta-decidim) during some period of time (a week at least), when they are considered ready, it is time for them to be merged into `master`:
|
70
|
+
|
71
|
+
1. Checkout the release stable branch `git checkout release/x.y-stable`.
|
72
|
+
1. Update `.decidim-version` by removing the `.rcN` suffix, leaving a clean version number like `x.y.z`
|
73
|
+
1. Run `bin/rake update_versions`, this will update all references to the new version.
|
74
|
+
1. Run `bin/rake bundle`, this will update all the `Gemfile.lock` files
|
75
|
+
1. Run `bin/rake webpack`, this will update the JavaScript bundle.
|
76
|
+
1. Commit all the changes: `git add . && git commit -m "Bump to v0.XX.0 final version" && git push origin release/x.y-stable`.
|
77
|
+
1. Create the PR for the new version.
|
78
|
+
1. `git checkout master && git checkout -b release/x.y.z`
|
79
|
+
1. `git merge release/x.y-stable`
|
80
|
+
1. `git checkout --theirs *`
|
81
|
+
1. `git checkout --theirs .github/* \.*`
|
82
|
+
1. Review changes in `CHANGELOG.md`, manually update and create a "changelog" commit if required.
|
83
|
+
1. `git push origin release/x.y.z`
|
84
|
+
1. Create the PR. The base for this PR should be `master`, but GitHub may crash if there are a lot of changes. As a workaround create the branch against `develop` and, when created, change the base to `master`.
|
85
|
+
1. Still don't merge it.
|
86
|
+
1. Before merging the PR to upgrade `master`, check that the stable branch for the previous version exists. For instance, if we are going to release v0.22.0, there should be a `release/0.21-stable` branch in the repository. If such branch does not exists, it has to be created now, before merging the new release. So, if this is the release of v0.22.0, branch off `release/0.21-stable` from `master`. These stable branches will be able to receive bugfixes, backports and will be the origin of patch releases for older releases.
|
87
|
+
1. Merge (after proper peer review) the PR to `master` and remove `release/x.y.z` branch.
|
88
|
+
1. Run `git checkout master && bin/rake release_all`, this will create all the tags, push the commits and tags and release the gems to RubyGems.
|
89
|
+
1. Once all the gems are published you should create a new release at this repository, just go to the [releases page](https://github.com/decidim/decidim/releases) and create a new one.
|
90
|
+
1. Create the stable branch for the current version. From `master`: `git checkout -b release/x.y-stable && git push origin release/x.y-stable`.
|
91
|
+
1. Update Decidim's Docker repository as explained in the Docker images section below.
|
92
|
+
1. Update Crowdin synchronization configuration with Github:
|
93
|
+
1. Add the new `release/x.y-stable` branch.
|
94
|
+
1. Remove from Crowdin branches that are not officially supported anymore. That way they don't synchronize with Github.
|
95
|
+
1. Update the `CHANGELOG.MD` in `release/x.y-stable`. At the top you should have an `Unreleased` header with the `Added`, `Changed`, `Fixed` and `Removed` empty sections. After that, the header with the current version. Add the `Unreleased` section or create the new current version section.
|
96
|
+
|
97
|
+
## Releasing patch versions
|
98
|
+
|
99
|
+
Releasing new versions from a ***release/x.y-stable*** branch is quite easy. The process is very similar from releasing a new Decidim version:
|
100
|
+
|
101
|
+
1. Checkout the branch you want to release: `git checkout -b release/x.y-stable`
|
102
|
+
1. Update `.decidim-version` to the new version number.
|
103
|
+
1. Run `bin/rake update_versions`, this will update all references to the new version.
|
104
|
+
1. Run `bin/rake bundle`, this will update all the `Gemfile.lock` files
|
105
|
+
1. Run `bin/rake webpack`, this will update the JavaScript bundle.
|
106
|
+
1. Update `CHANGELOG.MD`. At the top you should have an `Unreleased` header with the `Added`, `Changed`, `Fixed` and `Removed` empty sections. After that, the header with the current version and link like `## [0.20.0](https://github.com/decidim/decidim/tree/v0.20.0)` and again the headers for the `Added`, `Changed`, `Fixed` and `Removed` sections.
|
107
|
+
1. Commit all the changes: `git add . && git commit -m "Prepare VERSION release"`
|
108
|
+
1. Run `bin/rake release_all`, this will create all the tags, push the commits and tags and release the gems to RubyGems.
|
109
|
+
1. Once all the gems are published you should create a new release at this repository, just go to the [releases page](https://github.com/decidim/decidim/releases) and create a new one.
|
110
|
+
1. Update Decidim's Docker repository as explained in the Docker images section.
|
111
|
+
|
112
|
+
## Docker images for each release
|
113
|
+
|
114
|
+
1. After each release, you should update our [Docker repository](https://github.com/decidim/docker) so new images are build for the new release. To do it, just update `DECIDIM_VERSION` at [circle.yml](https://github.com/decidim/docker/blob/master/circle.yml).
|
@@ -0,0 +1,53 @@
|
|
1
|
+
# Share tokens
|
2
|
+
|
3
|
+
Share tokens can be assigned to any model to provide a system to share unpublished resources with expirable and manageable tokens.
|
4
|
+
|
5
|
+
A share token is created by a user with an expiration time, and can be added as a query param to access otherwise restricted locations.
|
6
|
+
|
7
|
+
## Add share tokens to a model
|
8
|
+
|
9
|
+
The model must `include Decidim::ShareableWithToken` and implement `shareable_url(share_token)`, which should return the public url for the resource you want to share, including the token as a query parameter.
|
10
|
+
|
11
|
+
```ruby
|
12
|
+
# Public: Public URL for your_resource with given share token as query parameter
|
13
|
+
def shareable_url(share_token)
|
14
|
+
your_resource_public_path(self, share_token: share_token.token)
|
15
|
+
end
|
16
|
+
```
|
17
|
+
|
18
|
+
## Set permissions
|
19
|
+
|
20
|
+
You should change permissions logic for the resource to check if there's a `share_token` query parameter in the request url, and call `Decidim::ShareToken.use!` to both check if the token is valid, and if it is, to *use it* (which increments `times_used` variable and sets `last_used_at` to current time).
|
21
|
+
|
22
|
+
It should do something similar to this:
|
23
|
+
|
24
|
+
```ruby
|
25
|
+
token = context[:share_token]
|
26
|
+
|
27
|
+
return unless token.present?
|
28
|
+
|
29
|
+
allow! if Decidim::ShareToken.use!(token_for: your_resource, token: token)
|
30
|
+
```
|
31
|
+
|
32
|
+
## Manage tokens
|
33
|
+
|
34
|
+
Render the partial `decidim-admin/app/views/decidim/admin/share_tokens/_share_tokens.html.erb` inside a view, with:
|
35
|
+
|
36
|
+
```ruby
|
37
|
+
locals: { share_tokens: your_share_tokens_variable }
|
38
|
+
```
|
39
|
+
|
40
|
+
to let admins see and manage tokens for that resource.
|
41
|
+
|
42
|
+
## Link to url with token
|
43
|
+
|
44
|
+
Implement a `share` action (see below) in the resource controller (admin scope), redirecting to a url with a newly generated token, so you can call `share_my_resource_url`.
|
45
|
+
|
46
|
+
```ruby
|
47
|
+
def share
|
48
|
+
@your_resource = YourResource.find(params[:id]) # or whatever
|
49
|
+
share_token = @your_resource.share_tokens.create!(user: current_user, organization: current_organization)
|
50
|
+
|
51
|
+
redirect_to share_token.url
|
52
|
+
end
|
53
|
+
```
|