moirai 0.1.0 → 0.2.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.github/workflows/test.yml +8 -4
- data/CHANGELOG.md +16 -0
- data/Gemfile +12 -1
- data/README.md +20 -2
- data/app/assets/config/moirai_manifest.js +0 -1
- data/app/assets/javascripts/moirai_translation_controller.js +32 -0
- data/app/controllers/moirai/translation_files_controller.rb +41 -16
- data/app/controllers/moirai/translations_controller.rb +7 -0
- data/app/models/moirai/change.rb +15 -0
- data/app/models/moirai/key_finder.rb +55 -0
- data/app/models/moirai/translation.rb +9 -5
- data/app/models/moirai/translation_dumper.rb +40 -13
- data/app/views/layouts/moirai/application.html.erb +0 -1
- data/app/views/moirai/translation_files/_form.html.erb +3 -3
- data/app/views/moirai/translation_files/index.html.erb +1 -1
- data/app/views/moirai/translation_files/show.html.erb +5 -5
- data/app/views/moirai/translations/index.html.erb +7 -0
- data/bin/check +3 -1
- data/config/routes.rb +2 -1
- data/lib/generators/moirai/install_generator.rb +1 -2
- data/lib/generators/moirai/migration_generator.rb +5 -1
- data/lib/generators/moirai/templates/make_moirai_translations_file_path_not_required.rb.erb +5 -0
- data/lib/i18n/backend/moirai.rb +1 -53
- data/lib/moirai/engine.rb +3 -6
- data/lib/moirai/pull_request_creator.rb +42 -33
- data/moirai.gemspec +1 -6
- metadata +9 -76
- data/app/assets/javascripts/moirai/application.js +0 -2
- data/app/assets/javascripts/moirai/controllers/moirai_translation_controller.js +0 -27
- data/app/assets/javascripts/moirai/stimulus/init.js +0 -5
- /data/lib/generators/moirai/templates/{migration.rb.erb → create_moirai_translations.rb.erb} +0 -0
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 5cd2eae769961645bc1baa4295ed58bf261beb38a48e934c1d3d38dd083f033a
|
4
|
+
data.tar.gz: fd3dddd2520c7f06987d814fec2986c5e02ac113748d39a1117315296cc0aaf3
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 5ae7c1af2f697ea2f7fad123f4c2451e3fa9ba86ffb048ae0b6846c2002bc532e50f762ecf000e32d76f6631d62ebd9f211288b4dca081e61e9d1bbcb05fd6ef
|
7
|
+
data.tar.gz: f113aa608adfe9b5fae4a073f2d23936bb30090b3fe9d544dfc7d9421663244a78e3350b7218f988fa9c413fe6aef6a6ab154dd79cb5d0160dd026205bdac30d
|
data/.github/workflows/test.yml
CHANGED
@@ -1,11 +1,12 @@
|
|
1
1
|
name: Test & lint
|
2
|
-
on: [push
|
2
|
+
on: [push]
|
3
3
|
|
4
4
|
env:
|
5
5
|
RAILS_ENV: test
|
6
6
|
PGHOST: localhost
|
7
7
|
PGUSER: postgres
|
8
|
-
|
8
|
+
MOIRAI_GITHUB_REPO_NAME: ${{ secrets.MOIRAI_GITHUB_REPO_NAME }}
|
9
|
+
MOIRAI_GITHUB_ACCESS_TOKEN: ${{ secrets.MOIRAI_GITHUB_ACCESS_TOKEN }}
|
9
10
|
jobs:
|
10
11
|
tests:
|
11
12
|
name: Test
|
@@ -14,7 +15,7 @@ jobs:
|
|
14
15
|
fail-fast: false
|
15
16
|
matrix:
|
16
17
|
os: [ubuntu-latest]
|
17
|
-
ruby: [3.
|
18
|
+
ruby: [3.3]
|
18
19
|
|
19
20
|
runs-on: ${{ matrix.os }}
|
20
21
|
|
@@ -32,7 +33,10 @@ jobs:
|
|
32
33
|
run: bundle install --jobs 4 --retry 3
|
33
34
|
|
34
35
|
- name: Run tests
|
35
|
-
run:
|
36
|
+
run: bundle exec rails test
|
37
|
+
|
38
|
+
- name: Run system tests
|
39
|
+
run: bundle exec rails test:system
|
36
40
|
|
37
41
|
|
38
42
|
lint:
|
data/CHANGELOG.md
CHANGED
@@ -1,6 +1,22 @@
|
|
1
|
+
## Unreleased
|
2
|
+
|
3
|
+
## 0.2.0
|
4
|
+
|
5
|
+
* Support for strings coming from gems ([@coorasse][])
|
6
|
+
* Support for new strings (not yet translated) ([@coorasse][])
|
7
|
+
|
8
|
+
## 0.1.1
|
9
|
+
|
10
|
+
* Review Stimulus controller ([@coorasse][])
|
11
|
+
|
1
12
|
## 0.1.0
|
2
13
|
|
3
14
|
* Gem structure created ([@oliveranthony17][])
|
15
|
+
* Database tables created ([@oliveranthony17][])
|
16
|
+
* Pull request creation ([@oliveranthony17][])
|
17
|
+
* Dummy app for tests ([@coorasse][])
|
18
|
+
* CRUD for translations ([@CuddlyBunion341][])
|
19
|
+
* Inline editing ([@CuddlyBunion341][])
|
4
20
|
|
5
21
|
[@coorasse]: https://github.com/coorasse
|
6
22
|
|
data/Gemfile
CHANGED
@@ -4,14 +4,25 @@ source "https://rubygems.org"
|
|
4
4
|
|
5
5
|
gemspec
|
6
6
|
|
7
|
+
gem "rails", "~> 7.2.0"
|
8
|
+
gem "octokit", ">= 4.0"
|
9
|
+
gem "importmap-rails"
|
10
|
+
gem "stimulus-rails"
|
11
|
+
|
7
12
|
group :development, :test do
|
8
13
|
gem "puma"
|
9
14
|
gem "sqlite3"
|
10
|
-
gem "minitest"
|
11
15
|
gem "rake"
|
16
|
+
gem "dotenv"
|
17
|
+
gem "minitest"
|
12
18
|
gem "standard"
|
13
19
|
gem "appraisal"
|
14
20
|
gem "better_errors"
|
15
21
|
gem "binding_of_caller"
|
16
22
|
gem "sprockets-rails"
|
17
23
|
end
|
24
|
+
|
25
|
+
group :test do
|
26
|
+
gem "capybara"
|
27
|
+
gem "selenium-webdriver"
|
28
|
+
end
|
data/README.md
CHANGED
@@ -46,6 +46,7 @@ Open a file, change the value of translations, and press ENTER to update the tra
|
|
46
46
|
### Inline editing
|
47
47
|
|
48
48
|
By default, inline editing is disabled. To enable it, set the `moirai=true` query parameter in the URL.
|
49
|
+
|
49
50
|
If you want to only allow specific users to perform inline editing, you can override the `moirai_edit_enabled?` method in your application helper.
|
50
51
|
|
51
52
|
```ruby
|
@@ -56,6 +57,18 @@ module ApplicationHelper
|
|
56
57
|
end
|
57
58
|
```
|
58
59
|
|
60
|
+
You also need to have the moirai_translations_controller.js Stimulus Controller initialized.
|
61
|
+
|
62
|
+
If you use importmaps:
|
63
|
+
|
64
|
+
Pin the controller in `config/importmap.rb`
|
65
|
+
|
66
|
+
```ruby
|
67
|
+
pin "controllers/moirai_translation_controller", to: "moirai_translation_controller.js"
|
68
|
+
```
|
69
|
+
|
70
|
+
If you’re unsure about all the possible configuration options, you can simply copy and paste the stimulus controller into your app as a fallback.
|
71
|
+
|
59
72
|
### Automatic PR creation with Octokit (**optional**)
|
60
73
|
|
61
74
|
If you would like Moirai to automatically create a pull request on GitHub to keep translations synchronized with the codebase,
|
@@ -153,6 +166,12 @@ end
|
|
153
166
|
|
154
167
|
4. Set your environment variables using the newly created `.env` file.
|
155
168
|
|
169
|
+
You will need a repository to test against and a token. Generate a new Fine-GRained Personal access token and give the necessary permissions to your repository.
|
170
|
+
See the image below as an example:
|
171
|
+
|
172
|
+
![](docs/github_settings.png)
|
173
|
+
|
174
|
+
|
156
175
|
5. Run the tests:
|
157
176
|
```bash
|
158
177
|
bin/check
|
@@ -169,8 +188,7 @@ end
|
|
169
188
|
* Support for interpolation
|
170
189
|
* Support for count variants
|
171
190
|
* Better inline editing tool
|
172
|
-
*
|
173
|
-
* Support for translations and strings coming from other gems
|
191
|
+
* Support for fallbacks: it should detect when a fallback string is in use and prevent attempts to override its value.
|
174
192
|
|
175
193
|
## License
|
176
194
|
The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
|
@@ -0,0 +1,32 @@
|
|
1
|
+
import { Controller } from "@hotwired/stimulus"
|
2
|
+
|
3
|
+
export default class MoiraiTranslationController extends Controller {
|
4
|
+
static values = {
|
5
|
+
key: String,
|
6
|
+
locale: String
|
7
|
+
}
|
8
|
+
|
9
|
+
click(event) {
|
10
|
+
event.preventDefault()
|
11
|
+
}
|
12
|
+
|
13
|
+
submit(event) {
|
14
|
+
const csrfToken = document.querySelector('meta[name="csrf-token"]').content
|
15
|
+
|
16
|
+
fetch('/moirai/translation_files', {
|
17
|
+
method: 'POST',
|
18
|
+
headers: {
|
19
|
+
'X-CSRF-Token': csrfToken,
|
20
|
+
'Content-Type': 'application/json',
|
21
|
+
'Accept': 'application/json'
|
22
|
+
},
|
23
|
+
body: JSON.stringify({
|
24
|
+
translation: {
|
25
|
+
key: this.keyValue,
|
26
|
+
locale: this.localeValue,
|
27
|
+
value: event.target.innerText
|
28
|
+
}
|
29
|
+
})
|
30
|
+
});
|
31
|
+
}
|
32
|
+
}
|
@@ -14,13 +14,13 @@ module Moirai
|
|
14
14
|
end
|
15
15
|
|
16
16
|
def show
|
17
|
-
@translation_keys = @file_handler.parse_file(@
|
17
|
+
@translation_keys = @file_handler.parse_file(@file_path)
|
18
|
+
@locale = @file_handler.get_first_key(@file_path)
|
19
|
+
@translations = Moirai::Translation.by_file_path(@file_path)
|
18
20
|
end
|
19
21
|
|
20
22
|
def create_or_update
|
21
|
-
if (translation = Translation.find_by(
|
22
|
-
key: translation_params[:key],
|
23
|
-
locale: @file_handler.get_first_key(translation_params[:file_path])))
|
23
|
+
if (translation = Translation.find_by(key: translation_params[:key], locale: translation_params[:locale]))
|
24
24
|
handle_update(translation)
|
25
25
|
else
|
26
26
|
handle_create
|
@@ -28,7 +28,7 @@ module Moirai
|
|
28
28
|
end
|
29
29
|
|
30
30
|
def open_pr
|
31
|
-
flash.notice = "I created an amazing
|
31
|
+
flash.notice = "I created an amazing Pull Request"
|
32
32
|
changes = Moirai::TranslationDumper.new.call
|
33
33
|
Moirai::PullRequestCreator.new.create_pull_request(changes)
|
34
34
|
redirect_back_or_to(root_path)
|
@@ -37,8 +37,7 @@ module Moirai
|
|
37
37
|
private
|
38
38
|
|
39
39
|
def handle_update(translation)
|
40
|
-
|
41
|
-
if translation_from_file[translation.key] == translation_params[:value] || translation_params[:value].blank?
|
40
|
+
if translation_params[:value].blank? || translation_same_as_current?
|
42
41
|
translation.destroy
|
43
42
|
flash.notice = "Translation #{translation.key} was successfully deleted."
|
44
43
|
redirect_to_translation_file(translation.file_path)
|
@@ -51,26 +50,36 @@ module Moirai
|
|
51
50
|
flash.alert = translation.errors.full_messages.join(", ")
|
52
51
|
end
|
53
52
|
|
54
|
-
|
53
|
+
success_response(translation)
|
55
54
|
end
|
56
55
|
|
57
56
|
def handle_create
|
58
|
-
|
59
|
-
if translation_from_file[translation_params[:key]] == translation_params[:value]
|
57
|
+
if translation_same_as_current?
|
60
58
|
flash.alert = "Translation #{translation_params[:key]} already exists."
|
61
|
-
|
59
|
+
redirect_back_or_to moirai_translation_files_path, status: :unprocessable_entity
|
62
60
|
return
|
63
61
|
end
|
64
62
|
|
65
63
|
translation = Translation.new(translation_params)
|
66
|
-
|
64
|
+
|
67
65
|
if translation.save
|
68
66
|
flash.notice = "Translation #{translation.key} was successfully created."
|
67
|
+
success_response(translation)
|
69
68
|
else
|
70
69
|
flash.alert = translation.errors.full_messages.join(", ")
|
70
|
+
redirect_back_or_to moirai_translation_files_path, status: :unprocessable_entity
|
71
71
|
end
|
72
|
+
end
|
72
73
|
|
73
|
-
|
74
|
+
def success_response(translation)
|
75
|
+
respond_to do |format|
|
76
|
+
format.json do
|
77
|
+
render json: {}
|
78
|
+
end
|
79
|
+
format.all do
|
80
|
+
redirect_to_translation_file(translation.file_path)
|
81
|
+
end
|
82
|
+
end
|
74
83
|
end
|
75
84
|
|
76
85
|
def redirect_to_translation_file(file_path)
|
@@ -78,16 +87,32 @@ module Moirai
|
|
78
87
|
end
|
79
88
|
|
80
89
|
def set_translation_file
|
81
|
-
@file_path = @file_handler.file_hashes[params[:
|
82
|
-
|
90
|
+
@file_path = @file_handler.file_hashes[params[:hashed_file_path]]
|
91
|
+
if @file_path.nil?
|
92
|
+
flash.alert = "File not found"
|
93
|
+
redirect_to moirai_translation_files_path, status: :not_found
|
94
|
+
end
|
83
95
|
end
|
84
96
|
|
85
97
|
def translation_params
|
86
|
-
params.require(:translation).permit(:key, :locale, :value
|
98
|
+
params.require(:translation).permit(:key, :locale, :value)
|
87
99
|
end
|
88
100
|
|
89
101
|
def load_file_handler
|
90
102
|
@file_handler = Moirai::TranslationFileHandler.new
|
91
103
|
end
|
104
|
+
|
105
|
+
# TODO: to resolve the last point of the TODOs we could look at the current translation (without moirai)
|
106
|
+
# I quickly tried but I need to use the original backend instead of the moirai one
|
107
|
+
# The problem is that if we set a value that is the same as currently being used via fallback,
|
108
|
+
# it will create an entry in the database, and afterwards will try to add it in the PR, which we don't want.
|
109
|
+
def translation_same_as_current?
|
110
|
+
file_paths = KeyFinder.new.file_paths_for(translation_params[:key], locale: translation_params[:locale])
|
111
|
+
|
112
|
+
return false if file_paths.empty?
|
113
|
+
return false unless file_paths.all? { |file_path| File.exist?(file_path) }
|
114
|
+
|
115
|
+
translation_params[:value] == @file_handler.parse_file(file_paths.first)[translation_params[:key]]
|
116
|
+
end
|
92
117
|
end
|
93
118
|
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Prepares a certain file to be changefd in a Pull Request.
|
4
|
+
# It takes care of adjusting the file_path and content
|
5
|
+
module Moirai
|
6
|
+
class Change
|
7
|
+
attr_reader :file_path, :content
|
8
|
+
|
9
|
+
def initialize(file_path, content)
|
10
|
+
@file_path = file_path
|
11
|
+
@file_path = file_path.to_s.start_with?("./") ? file_path : "./#{file_path}"
|
12
|
+
@content = content.to_s.end_with?("\n") ? content : "#{content}\n"
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Moirai
|
4
|
+
class KeyFinder
|
5
|
+
include I18n::Backend::Base
|
6
|
+
|
7
|
+
# Mutex to ensure that concurrent translations loading will be thread-safe
|
8
|
+
MUTEX = Mutex.new
|
9
|
+
|
10
|
+
def initialize
|
11
|
+
load_translations
|
12
|
+
end
|
13
|
+
|
14
|
+
# TODO: remove locale default
|
15
|
+
# Returns all the file_paths where the key is found, including gems.
|
16
|
+
def file_paths_for(key, locale: I18n.locale)
|
17
|
+
return [] if key.blank?
|
18
|
+
|
19
|
+
locale ||= I18n.locale
|
20
|
+
moirai_translations[locale.to_sym].select do |_filename, data|
|
21
|
+
data.dig(*key.split(".")).present?
|
22
|
+
end.map { |k, _| k }.sort { |file_path| file_path.start_with?(Rails.root.to_s) ? 0 : 1 }
|
23
|
+
end
|
24
|
+
|
25
|
+
def store_moirai_translations(filename, locale, data, options)
|
26
|
+
moirai_translations[locale] ||= Concurrent::Hash.new
|
27
|
+
|
28
|
+
locale = locale.to_sym
|
29
|
+
moirai_translations[locale] ||= Concurrent::Hash.new
|
30
|
+
moirai_translations[locale][filename] = data.with_indifferent_access
|
31
|
+
end
|
32
|
+
|
33
|
+
def moirai_translations(do_init: false)
|
34
|
+
@moirai_translations ||= Concurrent::Hash.new do |h, k|
|
35
|
+
MUTEX.synchronize do
|
36
|
+
h[k] = Concurrent::Hash.new
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def load_file(filename)
|
42
|
+
type = File.extname(filename).tr(".", "").downcase
|
43
|
+
raise I18n::UnknownFileType.new(type, filename) unless respond_to?(:"load_#{type}", true)
|
44
|
+
data, keys_symbolized = send(:"load_#{type}", filename)
|
45
|
+
unless data.is_a?(Hash)
|
46
|
+
raise I18n::InvalidLocaleData.new(filename, "expects it to return a hash, but does not")
|
47
|
+
end
|
48
|
+
data.each do |locale, d|
|
49
|
+
store_moirai_translations(filename, locale, d || {}, skip_symbolize_keys: keys_symbolized)
|
50
|
+
end
|
51
|
+
|
52
|
+
data
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
@@ -2,13 +2,17 @@
|
|
2
2
|
|
3
3
|
module Moirai
|
4
4
|
class Translation < Moirai::ApplicationRecord
|
5
|
-
validates_presence_of :key, :locale, :
|
6
|
-
validate :file_path_must_exist
|
5
|
+
validates_presence_of :key, :locale, :value
|
7
6
|
|
8
|
-
|
7
|
+
# what if the key is present in multiple file_paths?
|
8
|
+
def file_path
|
9
|
+
@key_finder ||= KeyFinder.new
|
10
|
+
@key_finder.file_paths_for(key, locale: locale).first
|
11
|
+
end
|
9
12
|
|
10
|
-
def
|
11
|
-
|
13
|
+
def self.by_file_path(file_path)
|
14
|
+
key_finder = KeyFinder.new
|
15
|
+
all.select { |translation| key_finder.file_paths_for(translation.key, locale: translation.locale).include?(file_path) }
|
12
16
|
end
|
13
17
|
end
|
14
18
|
end
|
@@ -1,24 +1,51 @@
|
|
1
1
|
module Moirai
|
2
2
|
class TranslationDumper
|
3
|
+
def initialize
|
4
|
+
@key_finder = KeyFinder.new
|
5
|
+
end
|
6
|
+
|
7
|
+
# @return Array[Moirai::Change]
|
3
8
|
def call
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
9
|
+
translations_by_file_path = group_translations_by_file_path
|
10
|
+
changes = []
|
11
|
+
translations_by_file_path.each do |file_path, translations|
|
12
|
+
relative_file_path = Pathname.new(file_path).relative_path_from(Rails.root)
|
13
|
+
changes << Change.new(relative_file_path, get_updated_file_contents(file_path, translations))
|
14
|
+
end
|
15
|
+
changes
|
16
|
+
end
|
17
|
+
|
18
|
+
def best_file_path_for(key, locale)
|
19
|
+
file_paths = @key_finder.file_paths_for(key, locale: locale)
|
20
|
+
file_paths.filter! { |p| p.start_with?(Rails.root.to_s) }
|
21
|
+
if file_paths.any?
|
22
|
+
file_paths.first
|
23
|
+
elsif key.split(".").size > 1
|
24
|
+
parent_key = key.split(".")[0..-2].join(".")
|
25
|
+
best_file_path_for(parent_key, locale)
|
26
|
+
else
|
27
|
+
"./config/locales/#{locale}.yml"
|
28
|
+
end
|
15
29
|
end
|
16
30
|
|
17
31
|
private
|
18
32
|
|
19
|
-
def
|
20
|
-
|
33
|
+
def group_translations_by_file_path
|
34
|
+
translations_grouped_by_file_path = {}
|
35
|
+
Moirai::Translation.order(created_at: :asc).each do |translation|
|
36
|
+
file_path = best_file_path_for(translation.key, translation.locale)
|
37
|
+
absolute_file_path = File.expand_path(file_path, Rails.root)
|
38
|
+
|
39
|
+
# skip file paths that don't belong to the project
|
40
|
+
next unless absolute_file_path.to_s.start_with?(Rails.root.to_s)
|
41
|
+
|
42
|
+
translations_grouped_by_file_path[absolute_file_path] ||= []
|
43
|
+
translations_grouped_by_file_path[absolute_file_path] << translation
|
44
|
+
end
|
45
|
+
translations_grouped_by_file_path
|
46
|
+
end
|
21
47
|
|
48
|
+
def get_updated_file_contents(file_path, translations)
|
22
49
|
yaml = YAML.load_file(file_path)
|
23
50
|
|
24
51
|
translations.each do |translation|
|
@@ -6,7 +6,6 @@
|
|
6
6
|
<%= csp_meta_tag %>
|
7
7
|
|
8
8
|
<%= stylesheet_link_tag "moirai/application", media: "all" %>
|
9
|
-
<%= javascript_include_tag "moirai/application", "data-turoblinks-track": "reload", type: "module" %>
|
10
9
|
<link rel="stylesheet" href="https://cdn.simplecss.org/simple.min.css">
|
11
10
|
<link rel="icon" href="">
|
12
11
|
</head>
|
@@ -1,9 +1,9 @@
|
|
1
1
|
<span
|
2
2
|
contenteditable
|
3
|
-
data-action="blur->moirai-translation#submit click->moirai-translation#click"
|
3
|
+
data-action="blur->moirai-translation#submit click->moirai-translation#click"
|
4
4
|
style="border: 1px dashed #1d9f74; min-width: 30px; display: inline-block;"
|
5
|
-
data-key="<%= key %>"
|
6
|
-
data-
|
5
|
+
data-moirai-translation-key-value="<%= key %>"
|
6
|
+
data-moirai-translation-locale-value="<%= locale %>"
|
7
7
|
data-controller="moirai-translation">
|
8
8
|
<%= value %>
|
9
9
|
</span>
|
@@ -2,7 +2,7 @@
|
|
2
2
|
<h1>Translation files</h1>
|
3
3
|
|
4
4
|
<% if Moirai::PullRequestCreator.available? %>
|
5
|
-
<%= button_to "Create or update
|
5
|
+
<%= button_to "Create or update Pull Request", moirai_open_pr_path %>
|
6
6
|
<% else %>
|
7
7
|
<p>PR creation is not available. Add the gem octokit to your gemfile to enable this feature</p>
|
8
8
|
<% end %>
|
@@ -13,7 +13,7 @@
|
|
13
13
|
</thead>
|
14
14
|
<tbody>
|
15
15
|
<% @translation_keys.each do |key, value| %>
|
16
|
-
<% translation =
|
16
|
+
<% translation = @translations.find { |t| t.key == key } %>
|
17
17
|
|
18
18
|
<tr>
|
19
19
|
<td>
|
@@ -24,10 +24,10 @@
|
|
24
24
|
<% end %>
|
25
25
|
</td>
|
26
26
|
<td>
|
27
|
-
<%= form_for Moirai::Translation.new, url: moirai_create_or_update_translation_path do |f| %>
|
28
|
-
<%= f.hidden_field :key
|
29
|
-
<%= f.hidden_field :
|
30
|
-
<%= f.text_field :value
|
27
|
+
<%= form_for translation&.presence || Moirai::Translation.new(key: key, locale: @locale, value: value), url: moirai_create_or_update_translation_path, method: :post do |f| %>
|
28
|
+
<%= f.hidden_field :key %>
|
29
|
+
<%= f.hidden_field :locale %>
|
30
|
+
<%= f.text_field :value %>
|
31
31
|
<%= f.submit 'Update', style: 'display: none;' %>
|
32
32
|
<% end %>
|
33
33
|
</td>
|
data/bin/check
CHANGED
data/config/routes.rb
CHANGED
@@ -3,7 +3,8 @@
|
|
3
3
|
Moirai::Engine.routes.draw do
|
4
4
|
root to: "translation_files#index"
|
5
5
|
|
6
|
-
resources :
|
6
|
+
resources :translations, only: %i[index]
|
7
|
+
resources :translation_files, only: %i[index show], as: "moirai_translation_files", param: :hashed_file_path
|
7
8
|
post "/translation_files/open_pr", to: "translation_files#open_pr", as: "moirai_open_pr"
|
8
9
|
post "/translation_files", to: "translation_files#create_or_update", as: "moirai_create_or_update_translation"
|
9
10
|
end
|
@@ -11,7 +11,11 @@ module Moirai
|
|
11
11
|
source_root File.expand_path("templates", __dir__)
|
12
12
|
|
13
13
|
def create_migration_file
|
14
|
-
migration_template "
|
14
|
+
migration_template "create_moirai_translations.rb.erb", "db/migrate/create_moirai_translations.rb", migration_version: migration_version
|
15
|
+
end
|
16
|
+
|
17
|
+
def file_path_migration_file
|
18
|
+
migration_template "make_moirai_translations_file_path_not_required.rb.erb", "db/migrate/make_moirai_translations_file_path_not_required.rb", migration_version: migration_version
|
15
19
|
end
|
16
20
|
|
17
21
|
private
|
data/lib/i18n/backend/moirai.rb
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
module I18n
|
2
2
|
module Backend
|
3
|
-
class Moirai < I18n::Backend::Simple
|
3
|
+
class Moirai < I18n::Backend::Simple
|
4
4
|
# TODO: mega inefficient. we don't want to perform a SQL query for each key!
|
5
5
|
def translate(locale, key, options = EMPTY_HASH)
|
6
6
|
overridden_translation = ::Moirai::Translation.find_by(locale: locale, key: key)
|
@@ -8,58 +8,6 @@ module I18n
|
|
8
8
|
overridden_translation.value
|
9
9
|
end
|
10
10
|
end
|
11
|
-
|
12
|
-
def store_moirai_translations(filename, locale, data, options)
|
13
|
-
moirai_translations[locale] ||= Concurrent::Hash.new
|
14
|
-
flatten_data = flatten_hash(filename, data)
|
15
|
-
flatten_data = Utils.deep_symbolize_keys(flatten_data) unless options.fetch(:skip_symbolize_keys, false)
|
16
|
-
Utils.deep_merge!(moirai_translations[locale], flatten_data)
|
17
|
-
end
|
18
|
-
|
19
|
-
def moirai_translations(do_init: false)
|
20
|
-
@moirai_translations ||= Concurrent::Hash.new do |h, k|
|
21
|
-
MUTEX.synchronize do
|
22
|
-
h[k] = Concurrent::Hash.new
|
23
|
-
end
|
24
|
-
end
|
25
|
-
end
|
26
|
-
|
27
|
-
def flatten_hash(filename, hash, parent_key = "", result = {})
|
28
|
-
hash.each do |key, value|
|
29
|
-
new_key = parent_key.empty? ? key.to_s : "#{parent_key}.#{key}"
|
30
|
-
case value
|
31
|
-
when Hash
|
32
|
-
flatten_hash(filename, value, new_key, result)
|
33
|
-
when Array
|
34
|
-
value.each_with_index do |item, index|
|
35
|
-
array_key = "#{new_key}.#{index}"
|
36
|
-
if item.is_a?(Hash)
|
37
|
-
flatten_hash(filename, item, array_key, result)
|
38
|
-
else
|
39
|
-
result[array_key] = filename
|
40
|
-
end
|
41
|
-
end
|
42
|
-
else
|
43
|
-
result[new_key] = filename
|
44
|
-
end
|
45
|
-
end
|
46
|
-
result
|
47
|
-
end
|
48
|
-
|
49
|
-
def load_file(filename)
|
50
|
-
type = File.extname(filename).tr(".", "").downcase
|
51
|
-
raise UnknownFileType.new(type, filename) unless respond_to?(:"load_#{type}", true)
|
52
|
-
data, keys_symbolized = send(:"load_#{type}", filename)
|
53
|
-
unless data.is_a?(Hash)
|
54
|
-
raise InvalidLocaleData.new(filename, "expects it to return a hash, but does not")
|
55
|
-
end
|
56
|
-
data.each do |locale, d|
|
57
|
-
store_translations(locale, d || {}, skip_symbolize_keys: keys_symbolized)
|
58
|
-
store_moirai_translations(filename, locale, d || {}, skip_symbolize_keys: keys_symbolized)
|
59
|
-
end
|
60
|
-
|
61
|
-
data
|
62
|
-
end
|
63
11
|
end
|
64
12
|
end
|
65
13
|
end
|
data/lib/moirai/engine.rb
CHANGED
@@ -9,9 +9,7 @@ module Moirai
|
|
9
9
|
end
|
10
10
|
|
11
11
|
config.after_initialize do
|
12
|
-
|
13
|
-
moirai_backend.eager_load!
|
14
|
-
I18n.backend = I18n::Backend::Chain.new(moirai_backend, I18n.backend)
|
12
|
+
I18n.backend = I18n::Backend::Chain.new(I18n::Backend::Moirai.new, I18n.backend)
|
15
13
|
end
|
16
14
|
|
17
15
|
# TODO: how to do this without rewriting the entire method?
|
@@ -25,12 +23,11 @@ module Moirai
|
|
25
23
|
value = original_translate(key, **options)
|
26
24
|
|
27
25
|
if moirai_edit_enabled?
|
28
|
-
|
29
|
-
file_path = moirai_translations[I18n.locale][scope_key_by_partial(key)]
|
26
|
+
@key_finder ||= Moirai::KeyFinder.new
|
30
27
|
|
31
28
|
render(partial: "moirai/translation_files/form",
|
32
29
|
locals: {key: scope_key_by_partial(key),
|
33
|
-
|
30
|
+
locale: I18n.locale,
|
34
31
|
value: value})
|
35
32
|
else
|
36
33
|
value
|
@@ -1,41 +1,43 @@
|
|
1
1
|
class Moirai::PullRequestCreator
|
2
|
-
BRANCH_NAME = "moirai-translations"
|
3
|
-
|
4
2
|
def self.available?
|
5
|
-
defined?(Octokit)
|
3
|
+
!!defined?(Octokit)
|
6
4
|
end
|
7
5
|
|
6
|
+
BRANCH_PREFIX = "moirai-translations-"
|
7
|
+
|
8
|
+
attr_reader :github_repo_name, :github_access_token, :github_client, :github_repository, :branch_name
|
9
|
+
|
8
10
|
def initialize
|
9
11
|
@github_repo_name = ENV["MOIRAI_GITHUB_REPO_NAME"] || Rails.application.credentials.dig(:moirai, :github_repo_name)
|
10
12
|
@github_access_token = ENV["MOIRAI_GITHUB_ACCESS_TOKEN"] || Rails.application.credentials.dig(:moirai, :github_access_token)
|
11
|
-
@
|
12
|
-
|
13
|
-
)
|
13
|
+
@github_client = Octokit::Client.new(access_token: github_access_token)
|
14
|
+
@github_repository = github_client.repo(github_repo_name)
|
14
15
|
end
|
15
16
|
|
16
|
-
|
17
|
-
|
18
|
-
|
17
|
+
# @param changes Array[Moirai::Change]
|
18
|
+
def create_pull_request(changes)
|
19
|
+
@branch_name = "#{BRANCH_PREFIX}#{Time.current.strftime("%F-%H-%M-%S")}-#{rand(1000..9999)}"
|
20
|
+
default_branch = github_repository.default_branch
|
19
21
|
|
20
|
-
if
|
21
|
-
|
22
|
+
if moirai_branch_exists?
|
23
|
+
Rails.logger.debug { "Branch #{branch_name} already exists - the branch will be updated with the new changes" }
|
22
24
|
else
|
23
|
-
|
24
|
-
default_branch_ref = @
|
25
|
+
Rails.logger.debug { "Branch #{branch_name} does not exist - creating branch" }
|
26
|
+
default_branch_ref = @github_client.ref(@github_repo_name, "heads/#{default_branch}")
|
25
27
|
latest_commit_sha = default_branch_ref.object.sha
|
26
28
|
|
27
|
-
@
|
29
|
+
@github_client.create_ref(@github_repo_name, "heads/#{branch_name}", latest_commit_sha)
|
28
30
|
end
|
29
31
|
|
30
|
-
|
31
|
-
update_file(
|
32
|
+
changes.each do |change|
|
33
|
+
update_file(change.file_path, change.content)
|
32
34
|
end
|
33
35
|
|
34
|
-
unless
|
35
|
-
pull_request = @
|
36
|
+
unless existing_open_pull_request.present?
|
37
|
+
pull_request = @github_client.create_pull_request(
|
36
38
|
@github_repo_name,
|
37
39
|
default_branch,
|
38
|
-
|
40
|
+
branch_name,
|
39
41
|
"Adding new content by Moirai",
|
40
42
|
"BODY - This is a pull request created by Moirai"
|
41
43
|
)
|
@@ -44,40 +46,47 @@ class Moirai::PullRequestCreator
|
|
44
46
|
end
|
45
47
|
end
|
46
48
|
|
47
|
-
def
|
48
|
-
@
|
49
|
+
def moirai_branch_exists?
|
50
|
+
@github_client.ref(@github_repo_name, "heads/#{branch_name}")
|
49
51
|
true
|
50
52
|
rescue Octokit::NotFound
|
51
53
|
false
|
52
54
|
end
|
53
55
|
|
56
|
+
def existing_open_pull_request
|
57
|
+
@github_client.pull_requests(@github_repo_name).find do |pull_request|
|
58
|
+
pull_request.head.ref.start_with?(BRANCH_PREFIX) && (pull_request.state == "open")
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
def cleanup
|
63
|
+
pr = existing_open_pull_request
|
64
|
+
@github_client.close_pull_request(@github_repo_name, pr.number)
|
65
|
+
@github_client.delete_branch(@github_repo_name, pr.head.ref)
|
66
|
+
end
|
67
|
+
|
68
|
+
private
|
69
|
+
|
54
70
|
def update_file(path, content)
|
55
71
|
# TODO: check what happens if branch exists
|
56
|
-
|
57
|
-
file = @client.contents(@github_repo_name, path: path, ref: BRANCH_NAME)
|
72
|
+
file = @github_client.contents(@github_repo_name, path: path, ref: branch_name)
|
58
73
|
file_sha = file.sha
|
59
74
|
|
60
|
-
@
|
75
|
+
@github_client.update_contents(
|
61
76
|
@github_repo_name,
|
62
77
|
path,
|
63
78
|
"Updating translations for #{path} by Moirai",
|
64
79
|
file_sha,
|
65
80
|
content,
|
66
|
-
branch:
|
81
|
+
branch: branch_name
|
67
82
|
)
|
68
83
|
rescue Octokit::NotFound
|
69
|
-
@
|
84
|
+
@github_client.create_contents(
|
70
85
|
@github_repo_name,
|
71
86
|
path,
|
72
87
|
"Creating translations for #{path} by Moirai",
|
73
88
|
content,
|
74
|
-
branch:
|
89
|
+
branch: branch_name
|
75
90
|
)
|
76
91
|
end
|
77
|
-
|
78
|
-
def pull_request_exists?
|
79
|
-
@client.pull_requests(@github_repo_name).any? do |pull_request|
|
80
|
-
pull_request.head.ref == BRANCH_NAME
|
81
|
-
end
|
82
|
-
end
|
83
92
|
end
|
data/moirai.gemspec
CHANGED
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
Gem::Specification.new do |spec|
|
4
4
|
spec.name = "moirai"
|
5
|
-
spec.version = "0.
|
5
|
+
spec.version = "0.2.0"
|
6
6
|
spec.authors = ["Alessandro Rodi", "Oliver Anthony", "Daniel Bengl"]
|
7
7
|
spec.email = %w[alessandro.rodi@renuo.ch oliver.anthony@renuo.ch daniel.bengl@renuo.ch]
|
8
8
|
|
@@ -25,11 +25,6 @@ Gem::Specification.new do |spec|
|
|
25
25
|
|
26
26
|
spec.required_ruby_version = ">= 2.7.0"
|
27
27
|
spec.add_dependency "rails", ">= 6.1"
|
28
|
-
spec.add_development_dependency "octokit", ">= 4.0"
|
29
|
-
spec.add_development_dependency "capybara"
|
30
|
-
spec.add_development_dependency "selenium-webdriver"
|
31
|
-
spec.add_development_dependency "rails", "~> 7.2.0"
|
32
|
-
spec.add_development_dependency "dotenv"
|
33
28
|
|
34
29
|
spec.license = "MIT"
|
35
30
|
spec.metadata["rubygems_mfa_required"] = "true"
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: moirai
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.2.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Alessandro Rodi
|
@@ -10,7 +10,7 @@ authors:
|
|
10
10
|
autorequire:
|
11
11
|
bindir: exe
|
12
12
|
cert_chain: []
|
13
|
-
date: 2024-
|
13
|
+
date: 2024-11-06 00:00:00.000000000 Z
|
14
14
|
dependencies:
|
15
15
|
- !ruby/object:Gem::Dependency
|
16
16
|
name: rails
|
@@ -26,76 +26,6 @@ dependencies:
|
|
26
26
|
- - ">="
|
27
27
|
- !ruby/object:Gem::Version
|
28
28
|
version: '6.1'
|
29
|
-
- !ruby/object:Gem::Dependency
|
30
|
-
name: octokit
|
31
|
-
requirement: !ruby/object:Gem::Requirement
|
32
|
-
requirements:
|
33
|
-
- - ">="
|
34
|
-
- !ruby/object:Gem::Version
|
35
|
-
version: '4.0'
|
36
|
-
type: :development
|
37
|
-
prerelease: false
|
38
|
-
version_requirements: !ruby/object:Gem::Requirement
|
39
|
-
requirements:
|
40
|
-
- - ">="
|
41
|
-
- !ruby/object:Gem::Version
|
42
|
-
version: '4.0'
|
43
|
-
- !ruby/object:Gem::Dependency
|
44
|
-
name: capybara
|
45
|
-
requirement: !ruby/object:Gem::Requirement
|
46
|
-
requirements:
|
47
|
-
- - ">="
|
48
|
-
- !ruby/object:Gem::Version
|
49
|
-
version: '0'
|
50
|
-
type: :development
|
51
|
-
prerelease: false
|
52
|
-
version_requirements: !ruby/object:Gem::Requirement
|
53
|
-
requirements:
|
54
|
-
- - ">="
|
55
|
-
- !ruby/object:Gem::Version
|
56
|
-
version: '0'
|
57
|
-
- !ruby/object:Gem::Dependency
|
58
|
-
name: selenium-webdriver
|
59
|
-
requirement: !ruby/object:Gem::Requirement
|
60
|
-
requirements:
|
61
|
-
- - ">="
|
62
|
-
- !ruby/object:Gem::Version
|
63
|
-
version: '0'
|
64
|
-
type: :development
|
65
|
-
prerelease: false
|
66
|
-
version_requirements: !ruby/object:Gem::Requirement
|
67
|
-
requirements:
|
68
|
-
- - ">="
|
69
|
-
- !ruby/object:Gem::Version
|
70
|
-
version: '0'
|
71
|
-
- !ruby/object:Gem::Dependency
|
72
|
-
name: rails
|
73
|
-
requirement: !ruby/object:Gem::Requirement
|
74
|
-
requirements:
|
75
|
-
- - "~>"
|
76
|
-
- !ruby/object:Gem::Version
|
77
|
-
version: 7.2.0
|
78
|
-
type: :development
|
79
|
-
prerelease: false
|
80
|
-
version_requirements: !ruby/object:Gem::Requirement
|
81
|
-
requirements:
|
82
|
-
- - "~>"
|
83
|
-
- !ruby/object:Gem::Version
|
84
|
-
version: 7.2.0
|
85
|
-
- !ruby/object:Gem::Dependency
|
86
|
-
name: dotenv
|
87
|
-
requirement: !ruby/object:Gem::Requirement
|
88
|
-
requirements:
|
89
|
-
- - ">="
|
90
|
-
- !ruby/object:Gem::Version
|
91
|
-
version: '0'
|
92
|
-
type: :development
|
93
|
-
prerelease: false
|
94
|
-
version_requirements: !ruby/object:Gem::Requirement
|
95
|
-
requirements:
|
96
|
-
- - ">="
|
97
|
-
- !ruby/object:Gem::Version
|
98
|
-
version: '0'
|
99
29
|
description: |-
|
100
30
|
This gem allows you to manage translation strings in real time,
|
101
31
|
viewing the live changes in the browser, with the changes then converted to a PR opened on the repository.
|
@@ -117,17 +47,18 @@ files:
|
|
117
47
|
- Rakefile
|
118
48
|
- app/assets/config/moirai_manifest.js
|
119
49
|
- app/assets/images/moirai/.keep
|
120
|
-
- app/assets/javascripts/
|
121
|
-
- app/assets/javascripts/moirai/controllers/moirai_translation_controller.js
|
122
|
-
- app/assets/javascripts/moirai/stimulus/init.js
|
50
|
+
- app/assets/javascripts/moirai_translation_controller.js
|
123
51
|
- app/assets/stylesheets/moirai/application.css
|
124
52
|
- app/controllers/concerns/.keep
|
125
53
|
- app/controllers/moirai/application_controller.rb
|
126
54
|
- app/controllers/moirai/translation_files_controller.rb
|
55
|
+
- app/controllers/moirai/translations_controller.rb
|
127
56
|
- app/helpers/moirai/application_helper.rb
|
128
57
|
- app/jobs/moirai/application_job.rb
|
129
58
|
- app/models/concerns/.keep
|
130
59
|
- app/models/moirai/application_record.rb
|
60
|
+
- app/models/moirai/change.rb
|
61
|
+
- app/models/moirai/key_finder.rb
|
131
62
|
- app/models/moirai/translation.rb
|
132
63
|
- app/models/moirai/translation_dumper.rb
|
133
64
|
- app/models/moirai/translation_file_handler.rb
|
@@ -135,6 +66,7 @@ files:
|
|
135
66
|
- app/views/moirai/translation_files/_form.html.erb
|
136
67
|
- app/views/moirai/translation_files/index.html.erb
|
137
68
|
- app/views/moirai/translation_files/show.html.erb
|
69
|
+
- app/views/moirai/translations/index.html.erb
|
138
70
|
- bin/check
|
139
71
|
- bin/console
|
140
72
|
- bin/fastcheck
|
@@ -143,7 +75,8 @@ files:
|
|
143
75
|
- config/routes.rb
|
144
76
|
- lib/generators/moirai/install_generator.rb
|
145
77
|
- lib/generators/moirai/migration_generator.rb
|
146
|
-
- lib/generators/moirai/templates/
|
78
|
+
- lib/generators/moirai/templates/create_moirai_translations.rb.erb
|
79
|
+
- lib/generators/moirai/templates/make_moirai_translations_file_path_not_required.rb.erb
|
147
80
|
- lib/i18n/backend/moirai.rb
|
148
81
|
- lib/moirai.rb
|
149
82
|
- lib/moirai/engine.rb
|
@@ -1,27 +0,0 @@
|
|
1
|
-
Stimulus.register(
|
2
|
-
"moirai-translation",
|
3
|
-
class extends Controller {
|
4
|
-
click(event) {
|
5
|
-
event.preventDefault()
|
6
|
-
}
|
7
|
-
|
8
|
-
submit(event) {
|
9
|
-
const {filePath, key} = event.target.dataset
|
10
|
-
|
11
|
-
const csrfToken = document.querySelector('meta[name="csrf-token"]').content
|
12
|
-
|
13
|
-
const formObject = new FormData()
|
14
|
-
formObject.append('translation[key]', key)
|
15
|
-
formObject.append('translation[file_path]', filePath)
|
16
|
-
formObject.append('translation[value]', event.target.innerText)
|
17
|
-
|
18
|
-
fetch(`/moirai/translation_files`, {
|
19
|
-
method: 'POST',
|
20
|
-
headers: {
|
21
|
-
'X-CSRF-Token': csrfToken
|
22
|
-
},
|
23
|
-
body: formObject
|
24
|
-
})
|
25
|
-
}
|
26
|
-
}
|
27
|
-
);
|
/data/lib/generators/moirai/templates/{migration.rb.erb → create_moirai_translations.rb.erb}
RENAMED
File without changes
|