lost_in_translation 0.2.18 → 0.2.19
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.gitignore +8 -0
- data/.rspec +2 -0
- data/CODE_OF_CONDUCT.md +49 -0
- data/Gemfile +6 -0
- data/LICENSE.txt +21 -0
- data/README.md +85 -0
- data/Rakefile +11 -0
- data/bin/console +14 -0
- data/bin/setup +8 -0
- data/lib/lit/difference.rb +41 -0
- data/lib/lit/file_functions.rb +101 -0
- data/lib/lit/hash.rb +31 -0
- data/lib/lit/railties.rb +13 -0
- data/lib/lit/user_interface.rb +115 -0
- data/lib/lost_in_translation.rb +58 -0
- data/lib/tasks/compare.rb +35 -0
- data/lib/tasks/count.rb +28 -0
- data/lib/tasks/delete_missing.rb +32 -0
- data/lib/tasks/fully_automatic.rb +33 -0
- data/lib/tasks/half_automatic.rb +33 -0
- data/lib/tasks/recent.rb +41 -0
- data/lib/tasks/sort_file.rb +25 -0
- data/lib/tasks/split_file.rb +37 -0
- data/lib/tasks/start.rb +59 -0
- data/lib/tasks/tasks.rake +46 -0
- data/lost_in_translation.gemspec +25 -0
- data/tmp/tmp_master.yml +0 -0
- data/tmp/tmp_master_copy.yml +0 -0
- data/tmp/tmp_slave.yml +0 -0
- metadata +31 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 0cd012a988d369d10222f9e41553e2c287f3db00
|
4
|
+
data.tar.gz: cc79df11ddca81bd6d222d837d761f68c61c7d6e
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 293c06e2bddb5c73360f639d9d889d85b9ee3571a0b7238d7ba200f1753053ae8ad722e2b8a6622d8fe4baf01c4b6c3183d8b5d89df881d7df15fff90787a5e7
|
7
|
+
data.tar.gz: bef7bf02e3417f17eba019d01ad07e46871ba4d6fefbb26ba29e5b3ef44b1fcfab8927b8f88b72dea3b4434066e0cc226ecd9a60185ef64a72c6abd89fb2e6e2
|
data/.gitignore
ADDED
data/.rspec
ADDED
data/CODE_OF_CONDUCT.md
ADDED
@@ -0,0 +1,49 @@
|
|
1
|
+
# Contributor Code of Conduct
|
2
|
+
|
3
|
+
As contributors and maintainers of this project, and in the interest of
|
4
|
+
fostering an open and welcoming community, we pledge to respect all people who
|
5
|
+
contribute through reporting issues, posting feature requests, updating
|
6
|
+
documentation, submitting pull requests or patches, and other activities.
|
7
|
+
|
8
|
+
We are committed to making participation in this project a harassment-free
|
9
|
+
experience for everyone, regardless of level of experience, gender, gender
|
10
|
+
identity and expression, sexual orientation, disability, personal appearance,
|
11
|
+
body size, race, ethnicity, age, religion, or nationality.
|
12
|
+
|
13
|
+
Examples of unacceptable behavior by participants include:
|
14
|
+
|
15
|
+
* The use of sexualized language or imagery
|
16
|
+
* Personal attacks
|
17
|
+
* Trolling or insulting/derogatory comments
|
18
|
+
* Public or private harassment
|
19
|
+
* Publishing other's private information, such as physical or electronic
|
20
|
+
addresses, without explicit permission
|
21
|
+
* Other unethical or unprofessional conduct
|
22
|
+
|
23
|
+
Project maintainers have the right and responsibility to remove, edit, or
|
24
|
+
reject comments, commits, code, wiki edits, issues, and other contributions
|
25
|
+
that are not aligned to this Code of Conduct, or to ban temporarily or
|
26
|
+
permanently any contributor for other behaviors that they deem inappropriate,
|
27
|
+
threatening, offensive, or harmful.
|
28
|
+
|
29
|
+
By adopting this Code of Conduct, project maintainers commit themselves to
|
30
|
+
fairly and consistently applying these principles to every aspect of managing
|
31
|
+
this project. Project maintainers who do not follow or enforce the Code of
|
32
|
+
Conduct may be permanently removed from the project team.
|
33
|
+
|
34
|
+
This code of conduct applies both within project spaces and in public spaces
|
35
|
+
when an individual is representing the project or its community.
|
36
|
+
|
37
|
+
Instances of abusive, harassing, or otherwise unacceptable behavior may be
|
38
|
+
reported by contacting a project maintainer at yves.goizet@hotmail.de. All
|
39
|
+
complaints will be reviewed and investigated and will result in a response that
|
40
|
+
is deemed necessary and appropriate to the circumstances. Maintainers are
|
41
|
+
obligated to maintain confidentiality with regard to the reporter of an
|
42
|
+
incident.
|
43
|
+
|
44
|
+
This Code of Conduct is adapted from the [Contributor Covenant][homepage],
|
45
|
+
version 1.3.0, available at
|
46
|
+
[http://contributor-covenant.org/version/1/3/0/][version]
|
47
|
+
|
48
|
+
[homepage]: http://contributor-covenant.org
|
49
|
+
[version]: http://contributor-covenant.org/version/1/3/0/
|
data/Gemfile
ADDED
data/LICENSE.txt
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
The MIT License (MIT)
|
2
|
+
|
3
|
+
Copyright (c) 2016 Datyv
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in
|
13
|
+
all copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
21
|
+
THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,85 @@
|
|
1
|
+
# LostInTranslation
|
2
|
+
|
3
|
+
This gem provides you three tasks to translate your locale yamls!
|
4
|
+
|
5
|
+
## Installation
|
6
|
+
|
7
|
+
Add this line to your application's Gemfile:
|
8
|
+
|
9
|
+
```ruby
|
10
|
+
gem 'lost_in_translation'
|
11
|
+
```
|
12
|
+
|
13
|
+
And then execute:
|
14
|
+
|
15
|
+
$ bundle
|
16
|
+
|
17
|
+
Or install it yourself as:
|
18
|
+
|
19
|
+
$ gem install lost_in_translation
|
20
|
+
|
21
|
+
## Usage
|
22
|
+
There are three tasks you can execute.
|
23
|
+
|
24
|
+
### ATTENTION: Use at your own risk! Back your shit up, before using this!
|
25
|
+
Each will ask you for the file
|
26
|
+
If this gem is installed inside a Rails-Application you just have to written
|
27
|
+
e.g.:
|
28
|
+
```
|
29
|
+
'en' -> for en.yml
|
30
|
+
'de' -> for de.yml
|
31
|
+
...
|
32
|
+
```
|
33
|
+
One will be the master-file (this one will not be changed)
|
34
|
+
and
|
35
|
+
One will be the slave-file (this one will be changed)
|
36
|
+
|
37
|
+
#### Script 1: Interactive
|
38
|
+
```bash
|
39
|
+
rake lit:interactive
|
40
|
+
```
|
41
|
+
This script will ask you for the missing value.
|
42
|
+
It will find every difference and will ask you to fill every hole... yeah.
|
43
|
+
After everything is filled out it will merge the data back into your File.
|
44
|
+
|
45
|
+
#### Script 2: Recent
|
46
|
+
```bash
|
47
|
+
rake lit:recent
|
48
|
+
```
|
49
|
+
REQUIRED: Connection to Git
|
50
|
+
This script will ask you for the missing value.
|
51
|
+
It will find everything you changed in your master-file since you last pushed commit.
|
52
|
+
After everything is filled out it will merge the data back into your File.
|
53
|
+
|
54
|
+
#### Script 3: Cleanup (Deletes something!)
|
55
|
+
```bash
|
56
|
+
rake lit:cleanup
|
57
|
+
```
|
58
|
+
This script will find every key/value inside your slave-file that isn´t present in your master-file.
|
59
|
+
And it will be deleted!!!!
|
60
|
+
|
61
|
+
#### Sometimes the Rake tasks aren't working
|
62
|
+
In this case you can use the Rails-Console and the Class -LostInTranslation-.
|
63
|
+
|
64
|
+
```bash
|
65
|
+
rails c
|
66
|
+
```
|
67
|
+
```
|
68
|
+
LostInTranslation.interactive
|
69
|
+
or
|
70
|
+
LostInTranslation.recent
|
71
|
+
or
|
72
|
+
LostInTranslation.cleanup
|
73
|
+
```
|
74
|
+
|
75
|
+
#### Yet to come
|
76
|
+
I am planning on an integration of Translation APIs.
|
77
|
+
|
78
|
+
## Contributing
|
79
|
+
|
80
|
+
Bug reports and pull requests are welcome on GitHub at https://github.com/datyv/lost_in_translation. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct.
|
81
|
+
|
82
|
+
|
83
|
+
## License
|
84
|
+
|
85
|
+
The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).
|
data/Rakefile
ADDED
data/bin/console
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
|
3
|
+
require "bundler/setup"
|
4
|
+
require "lost_in_translation"
|
5
|
+
|
6
|
+
# You can add fixtures and/or initialization code here to make experimenting
|
7
|
+
# with your gem easier. You can also use a different console, if you like.
|
8
|
+
|
9
|
+
# (If you use this, don't forget to add pry to your Gemfile!)
|
10
|
+
# require "pry"
|
11
|
+
# Pry.start
|
12
|
+
|
13
|
+
require "irb"
|
14
|
+
IRB.start
|
data/bin/setup
ADDED
@@ -0,0 +1,41 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
#
|
4
|
+
module LostInTranslation
|
5
|
+
def self.diff(root, compared, lang1, lang2, structure = [], new_array = [], options = {})
|
6
|
+
@slave_file = options[:slave_file] || ''
|
7
|
+
@mode = options[:mode] || 'count' # replace, clean, count
|
8
|
+
@diff_count = options[:diff_count] || 0
|
9
|
+
@diff_counter = options[:diff_counter] || 0
|
10
|
+
root.each_pair do |key, value|
|
11
|
+
next_root = root[key]
|
12
|
+
next_compared = compared.nil? ? nil : compared[key]
|
13
|
+
new_structure = structure.dup << key
|
14
|
+
if value.is_a?(String) && (compared.nil? || compared[key].nil?)
|
15
|
+
if value_missing(@slave_file, new_structure)
|
16
|
+
@diff_counter += 1
|
17
|
+
if @mode == 'replace'
|
18
|
+
new_val = ask_for_translation(value, new_structure, lang1, lang2, @diff_count, @diff_counter)
|
19
|
+
new_array << new_val unless new_val.blank?
|
20
|
+
elsif @mode == 'clean'
|
21
|
+
new_array << "#{new_structure.join('---')}---#{value}"
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
25
|
+
if next_root.is_a? Hash
|
26
|
+
diff(next_root, next_compared, lang1, lang2, new_structure, new_array,
|
27
|
+
diff_counter: @diff_counter, diff_count: @diff_count, mode: @mode, slave_file: @slave_file)
|
28
|
+
end
|
29
|
+
end
|
30
|
+
@mode == 'count' ? @diff_counter : new_array
|
31
|
+
end
|
32
|
+
|
33
|
+
def self.value_missing(slave_file, structure)
|
34
|
+
return true if slave_file.blank?
|
35
|
+
slave_file.dig(*structure).blank?
|
36
|
+
end
|
37
|
+
|
38
|
+
def self.get_locale(path)
|
39
|
+
path.split('/').last.split('.').first unless path.empty?
|
40
|
+
end
|
41
|
+
end
|
@@ -0,0 +1,101 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
#
|
4
|
+
module LostInTranslation
|
5
|
+
def self.prepare_for_edit(options = {})
|
6
|
+
paths = tmp_paths
|
7
|
+
FileUtils.cp(options[:master], paths[:master]) if options[:master]
|
8
|
+
FileUtils.cp(options[:slave], paths[:slave]) if options[:slave]
|
9
|
+
end
|
10
|
+
|
11
|
+
def self.prepare_paths(paths = [])
|
12
|
+
paths.each do |p|
|
13
|
+
prepare_yaml(p, false)
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
def self.postpare_paths(paths = [])
|
18
|
+
paths.each do |p|
|
19
|
+
prepare_yaml(p, true)
|
20
|
+
end
|
21
|
+
end
|
22
|
+
|
23
|
+
def self.define_snippets
|
24
|
+
snippets = []
|
25
|
+
snippets << ['<<', 'a_greater_than_sign']
|
26
|
+
snippets << ['*', 'an_asterik_sign']
|
27
|
+
snippets << ['!', 'a_bang_sign']
|
28
|
+
snippets << ['%', 'a_percentage_sign']
|
29
|
+
snippets << ['a_bang_sign \'', '\'a_bang_sign']
|
30
|
+
snippets
|
31
|
+
end
|
32
|
+
|
33
|
+
def self.reset_tmp_files
|
34
|
+
paths = tmp_paths
|
35
|
+
paths.each_value do |path|
|
36
|
+
File.open(path, 'w') { |file| file.write('') }
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
private
|
41
|
+
|
42
|
+
def self.prepare_yaml(path, post)
|
43
|
+
text = File.read(path)
|
44
|
+
snippets = define_snippets
|
45
|
+
|
46
|
+
snippets = snippets.reverse if post
|
47
|
+
snippets.each do |snippet|
|
48
|
+
snippet = snippet.reverse if post
|
49
|
+
text = text.gsub(snippet.first, snippet.second)
|
50
|
+
end
|
51
|
+
text = post ? postpare(snippets, text) : prepare(snippets, text)
|
52
|
+
|
53
|
+
File.open(path, 'w') { |file| file.puts text }
|
54
|
+
end
|
55
|
+
|
56
|
+
def self.postpare(variables, text)
|
57
|
+
variables = text.scan(/varyberryterry.*/)
|
58
|
+
variables.each do |var|
|
59
|
+
new_var = var.delete(':')
|
60
|
+
new_var = new_var.gsub('varyberryterry', ': &')
|
61
|
+
text = text.gsub(var, new_var)
|
62
|
+
end
|
63
|
+
text = add_single_quotes(text)
|
64
|
+
text = text.gsub(': !,', ": ! ','")
|
65
|
+
text = text.gsub('\'!', '! \'')
|
66
|
+
text = text.gsub(/---\n/, '')
|
67
|
+
text = text.gsub('!…', "! '…'")
|
68
|
+
text = text.gsub('!:distance_in_words', "! ':distance_in_words'")
|
69
|
+
text
|
70
|
+
end
|
71
|
+
|
72
|
+
def self.prepare(variables, text)
|
73
|
+
variables = text.scan(/: &.*/)
|
74
|
+
variables.each do |var|
|
75
|
+
new_var = var.gsub(': &', 'varyberryterry')
|
76
|
+
new_var += ':'
|
77
|
+
text = text.gsub(var, new_var)
|
78
|
+
end
|
79
|
+
text
|
80
|
+
end
|
81
|
+
|
82
|
+
def self.add_single_quotes(text)
|
83
|
+
tmp = ''
|
84
|
+
text.each_line do |line|
|
85
|
+
tmp += edit_line(line)
|
86
|
+
end
|
87
|
+
tmp
|
88
|
+
end
|
89
|
+
|
90
|
+
def self.edit_line(line)
|
91
|
+
return line if line.match(/: .*%.*$/).blank?
|
92
|
+
arr = line.split(': ', 2)
|
93
|
+
return line if arr[1].blank?
|
94
|
+
arr[1] = arr[1].gsub("\'", '')
|
95
|
+
arr[1] = arr[1].gsub("\"", '')
|
96
|
+
arr[1] = "'" + arr[1] unless arr[1].start_with?("\'") || arr[1].start_with?("\"")
|
97
|
+
arr[1] = arr[1].squish + "'\n" unless arr[1].squish.end_with?("\'") || arr[1].squish.end_with?("\"")
|
98
|
+
arr[1] = arr[1].gsub("\'! ", "! \'")
|
99
|
+
return arr.join(': ')
|
100
|
+
end
|
101
|
+
end
|
data/lib/lit/hash.rb
ADDED
@@ -0,0 +1,31 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
#
|
4
|
+
module LostInTranslation
|
5
|
+
def self.string_to_hash(string)
|
6
|
+
array = string.split('---')
|
7
|
+
value = array.pop
|
8
|
+
arr = array.reverse
|
9
|
+
arr[1..-1].inject(arr[0] => value) { |memo, i| { i => memo } }
|
10
|
+
end
|
11
|
+
|
12
|
+
def self.merge_hash(merge_from, merge_to)
|
13
|
+
return if merge_from.is_a?(String) || merge_to.is_a?(String)
|
14
|
+
merged_hash = merge_to
|
15
|
+
first_key = merge_from.keys.first
|
16
|
+
merged_hash[first_key] = if merge_to.key?(first_key)
|
17
|
+
merge_hash(merge_from[first_key], merge_to[first_key])
|
18
|
+
else
|
19
|
+
merge_from[first_key]
|
20
|
+
end
|
21
|
+
merged_hash
|
22
|
+
end
|
23
|
+
|
24
|
+
def self.sort_hash(object)
|
25
|
+
return object unless object.is_a?(Hash)
|
26
|
+
hash = {}
|
27
|
+
object.each { |k, v| hash[k] = sort_hash(v) }
|
28
|
+
sorted = hash.sort { |a, b| a[0].to_s <=> b[0].to_s }
|
29
|
+
hash.class[sorted]
|
30
|
+
end
|
31
|
+
end
|
data/lib/lit/railties.rb
ADDED
@@ -0,0 +1,115 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
#
|
4
|
+
module LostInTranslation
|
5
|
+
require 'readline'
|
6
|
+
|
7
|
+
def self.ask_for_file(master, namespace = '')
|
8
|
+
text = master ? "# Master-Locale (e.g. 'en'): " : "# Slave-Locale (e.g. 'de'): "
|
9
|
+
file = {}
|
10
|
+
file[:lang] = [(print text), Readline.readline()][1]
|
11
|
+
file[:filename] = namespace.blank? ? file[:lang] : "#{namespace}.#{file[:lang]}"
|
12
|
+
file[:path] = defined?(Rails) ? "#{Rails.root}/config/locales/#{file[:filename]}.yml" : ask_for_path
|
13
|
+
if !File.exist?(file[:path]) && master
|
14
|
+
log "File #{file[:path]} not found!"
|
15
|
+
file[:path] = ask_for_path
|
16
|
+
elsif !File.exist?(file[:path]) && !master
|
17
|
+
log "File #{file[:filename]} not found! It will be created!"
|
18
|
+
File.open(file[:path], 'w') { |f| f.write({ file[:lang] => {} }.to_yaml(line_width: -1)) }
|
19
|
+
end
|
20
|
+
file[:app_name] = Rails&.application&.class&.parent_name
|
21
|
+
file
|
22
|
+
end
|
23
|
+
|
24
|
+
def self.ask_for_locales_path
|
25
|
+
return "#{Rails.root}/config/locales" if defined?(Rails)
|
26
|
+
log
|
27
|
+
log 'Please type in the absolute path your locales-folder?'
|
28
|
+
log 'e.g. /home/youruser/Documents/your-awesome-app/config/locales'
|
29
|
+
path = [(print '# '), Readline.readline()][1]
|
30
|
+
path
|
31
|
+
end
|
32
|
+
|
33
|
+
def self.ask_for_permission(master, count, slave = nil)
|
34
|
+
log
|
35
|
+
log 'Comparing Locales'
|
36
|
+
log "Application: #{master[:app_name]}" if master[:app_name]
|
37
|
+
log "Master Locale: #{master[:filename]}.yml" if master
|
38
|
+
log "Slave Locale: #{slave[:filename]}.yml" if slave
|
39
|
+
log "Difference: #{count} Entries"
|
40
|
+
a = [(print "# Wanna do it? [Y/n]: "), Readline.readline()][1]
|
41
|
+
a == 'y' || a == 'Y' || a == ''
|
42
|
+
end
|
43
|
+
|
44
|
+
def self.ask_for_task(tasks)
|
45
|
+
log
|
46
|
+
log 'Welcome, stranger!'
|
47
|
+
log 'Are you lost in translation, too?'
|
48
|
+
log 'Then I will be glad to help you one of the following tasks!'
|
49
|
+
tasks.each_with_index do |task, i|
|
50
|
+
log "#{i + 1} => #{task[:text]}"
|
51
|
+
end
|
52
|
+
log 'Soo ... what can I do for you? (Type in the Number)'
|
53
|
+
id = [(print 'Task No.: '), Readline.readline()][1]
|
54
|
+
tasks[(id.to_i - 1)]
|
55
|
+
rescue
|
56
|
+
puts "Cannot find task with id #{id}"
|
57
|
+
end
|
58
|
+
|
59
|
+
def self.ask_for_split(file, count)
|
60
|
+
log
|
61
|
+
log "Splitting File #{file}.yml"
|
62
|
+
log "File-Count: #{count} new files will be create!"
|
63
|
+
a = [(print "# Wanna do it? [Y/n]: "), Readline.readline()][1]
|
64
|
+
a == 'y' || a == 'Y' || a == ''
|
65
|
+
end
|
66
|
+
|
67
|
+
def self.ask_for_sorting(file, app_name)
|
68
|
+
log
|
69
|
+
log "Application: #{app_name}" unless app_name.blank?
|
70
|
+
log "Do you want the #{file}.yml to be sorted?"
|
71
|
+
log 'Alphabetically & Recursive (ASC)'
|
72
|
+
a = [(print "# [Y/n]: "), Readline.readline()][1]
|
73
|
+
a == 'y' || a == 'Y' || a == ''
|
74
|
+
end
|
75
|
+
|
76
|
+
def self.ask_for_translation(value, new_structure, lang1, lang2, diff_count, diff_counter)
|
77
|
+
system 'clear'
|
78
|
+
log
|
79
|
+
log "#{diff_counter} / #{diff_count}"
|
80
|
+
new_s = new_structure.join('---')
|
81
|
+
log "#{new_s} is missing"
|
82
|
+
|
83
|
+
snippets = LostInTranslation.define_snippets.reverse
|
84
|
+
snippets.each do |snippet|
|
85
|
+
value = value.gsub(snippet.second, snippet.first)
|
86
|
+
end
|
87
|
+
|
88
|
+
puts "# In #{lang1.upcase}: #{value}"
|
89
|
+
new_val = [(print "# In #{lang2.upcase}: "), Readline.readline()][1]
|
90
|
+
new_val = (new_s + '---' + new_val) unless new_val.blank?
|
91
|
+
new_val
|
92
|
+
end
|
93
|
+
|
94
|
+
def self.log(text = '')
|
95
|
+
max_length = 80
|
96
|
+
outside = 4
|
97
|
+
puts '#' * max_length if text.blank?
|
98
|
+
arr = text.scan(/.{1,#{max_length - outside}}/)
|
99
|
+
arr.each do |str|
|
100
|
+
fill_length = max_length - str.length - outside
|
101
|
+
fill = ' ' * fill_length
|
102
|
+
puts '# ' + str + fill + ' #'
|
103
|
+
end
|
104
|
+
end
|
105
|
+
|
106
|
+
private
|
107
|
+
|
108
|
+
def self.ask_for_path
|
109
|
+
log
|
110
|
+
log 'Please type in the absolute path your locale?'
|
111
|
+
log 'e.g. /home/youruser/Documents/your-awesome-app/config/locales/de.yml'
|
112
|
+
path = [(print '# '), Readline.readline()][1]
|
113
|
+
path
|
114
|
+
end
|
115
|
+
end
|
@@ -0,0 +1,58 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'lit/difference'
|
4
|
+
require 'lit/file_functions'
|
5
|
+
require 'lit/hash'
|
6
|
+
require 'lit/user_interface'
|
7
|
+
require 'lit/railties'
|
8
|
+
|
9
|
+
require 'tasks/compare'
|
10
|
+
require 'tasks/delete_missing'
|
11
|
+
require 'tasks/fully_automatic'
|
12
|
+
require 'tasks/half_automatic'
|
13
|
+
require 'tasks/recent'
|
14
|
+
require 'tasks/sort_file'
|
15
|
+
require 'tasks/split_file'
|
16
|
+
require 'tasks/start'
|
17
|
+
require 'tasks/count'
|
18
|
+
|
19
|
+
module LostInTranslation
|
20
|
+
def self.root
|
21
|
+
File.dirname __dir__
|
22
|
+
end
|
23
|
+
|
24
|
+
def self.tmp_paths
|
25
|
+
tmp_paths = {}
|
26
|
+
tmp_paths[:copy] = "#{root}/tmp/tmp_master_copy.yml"
|
27
|
+
tmp_paths[:master] = "#{root}/tmp/tmp_master.yml"
|
28
|
+
tmp_paths[:slave] = "#{root}/tmp/tmp_slave.yml"
|
29
|
+
tmp_paths
|
30
|
+
end
|
31
|
+
|
32
|
+
def self.init(with_slave = true)
|
33
|
+
system 'clear'
|
34
|
+
log
|
35
|
+
log 'What I18n-yaml files do you want to compare?'
|
36
|
+
namespace = [(print "# Namespace? ('namespace' for 'namespace.en.yml'): "), Readline.readline()][1]
|
37
|
+
master = ask_for_file(true, namespace) # lang path app_name
|
38
|
+
slave = with_slave ? ask_for_file(false, namespace) : {}
|
39
|
+
invalid_filepath if !File.exist?(master[:path]) && (!File.exist?(slave[:path]) && with_slave)
|
40
|
+
[master, slave, tmp_paths]
|
41
|
+
end
|
42
|
+
|
43
|
+
def self.breakout
|
44
|
+
raise "Well in that case, forget it!"
|
45
|
+
end
|
46
|
+
|
47
|
+
def self.invalid_filepath
|
48
|
+
raise "One on the filepaths are invalid!"
|
49
|
+
end
|
50
|
+
|
51
|
+
def self.tbd
|
52
|
+
raise "This feature is not yet implemented! Please take another!"
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
class Lit
|
57
|
+
extend LostInTranslation
|
58
|
+
end
|
@@ -0,0 +1,35 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.compare
|
5
|
+
@master, @slave, paths = init
|
6
|
+
|
7
|
+
prepare_for_edit(master: @master[:path], slave: @slave[:path])
|
8
|
+
prepare_paths([paths[:master], paths[:slave]])
|
9
|
+
|
10
|
+
master_file = YAML.load_file(paths[:master])
|
11
|
+
slave_file = YAML.load_file(paths[:slave])
|
12
|
+
|
13
|
+
count = diff(master_file[@master[:lang]], slave_file[@slave[:lang]], @master[:lang],
|
14
|
+
@slave[:lang], [@slave[:lang]], [], mode: 'count')
|
15
|
+
|
16
|
+
breakout unless ask_for_permission(@master, count, @slave)
|
17
|
+
|
18
|
+
new_strings_array = diff(master_file[@master[:lang]], slave_file[@slave[:lang]], @master[:lang],
|
19
|
+
@slave[:lang], [@slave[:lang]], [], diff_count: count, mode: 'replace')
|
20
|
+
|
21
|
+
new_strings_array.each do |string|
|
22
|
+
result = string_to_hash(string)
|
23
|
+
merge_hash(result, slave_file)
|
24
|
+
end
|
25
|
+
|
26
|
+
File.open(paths[:slave], 'w') { |file| file.write(slave_file.to_yaml(line_width: -1)) }
|
27
|
+
postpare_paths([paths[:slave]])
|
28
|
+
FileUtils.cp(paths[:slave], @slave[:path])
|
29
|
+
|
30
|
+
reset_tmp_files
|
31
|
+
|
32
|
+
log 'Done'
|
33
|
+
log
|
34
|
+
end
|
35
|
+
end
|
data/lib/tasks/count.rb
ADDED
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.count
|
5
|
+
@master, @slave, paths = init
|
6
|
+
|
7
|
+
prepare_for_edit(master: @master[:path], slave: @slave[:path])
|
8
|
+
prepare_paths([paths[:master], paths[:slave]])
|
9
|
+
|
10
|
+
master_file = YAML.load_file(paths[:master])
|
11
|
+
slave_file = YAML.load_file(paths[:slave])
|
12
|
+
|
13
|
+
count = diff(master_file[@master[:lang]], slave_file[@slave[:lang]], @master[:lang], @slave[:lang],
|
14
|
+
[@slave[:lang]], [], mode: 'count')
|
15
|
+
|
16
|
+
count2 = diff(slave_file[@slave[:lang]], master_file[@master[:lang]], @slave[:lang], @master[:lang],
|
17
|
+
[@master[:lang]], [], mode: 'count')
|
18
|
+
|
19
|
+
|
20
|
+
reset_tmp_files
|
21
|
+
log
|
22
|
+
log 'Differences between Locales'
|
23
|
+
log "#{@master[:filename]}.yml => #{@slave[:filename]}.yml = #{count}"
|
24
|
+
log "#{@slave[:filename]}.yml => #{@master[:filename]}.yml = #{count2}"
|
25
|
+
log 'Done'
|
26
|
+
log
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,32 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.delete_missing
|
5
|
+
tbd
|
6
|
+
@master, @slave, paths = init
|
7
|
+
|
8
|
+
prepare_for_edit(master: @master[:path], slave: @slave[:path])
|
9
|
+
prepare_paths([paths[:master], paths[:slave]])
|
10
|
+
|
11
|
+
master_file = YAML.load_file(paths[:master])
|
12
|
+
slave_file = YAML.load_file(paths[:slave])
|
13
|
+
|
14
|
+
breakout unless ask_for_permission(@master, count, @slave)
|
15
|
+
|
16
|
+
new_strings_array = diff(master_file[@master[:lang]], slave_file[@slave[:lang]], @master[:lang],
|
17
|
+
@slave[:lang], [@slave[:lang]], [], mode: 'clean')
|
18
|
+
|
19
|
+
master_file = {}
|
20
|
+
new_strings_array.each do |string|
|
21
|
+
string = string.split('---').map { |x| x == @slave[:lang] ? @master[:lang] : x }.join('---')
|
22
|
+
result = string_to_hash(string)
|
23
|
+
merge_hash(result, master_file)
|
24
|
+
end
|
25
|
+
|
26
|
+
File.open(@master[:path], 'w') { |file| file.write(master_file.to_yaml) }
|
27
|
+
postpare_paths([@master[:path], @slave[:path]])
|
28
|
+
reset_tmp_files
|
29
|
+
log 'Done'
|
30
|
+
log
|
31
|
+
end
|
32
|
+
end
|
@@ -0,0 +1,33 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.fully_automatic
|
5
|
+
tbd
|
6
|
+
@master, @slave, paths = init
|
7
|
+
|
8
|
+
prepare_for_edit(master: @master[:path], slave: @slave[:path])
|
9
|
+
prepare_paths([paths[:master], paths[:slave]])
|
10
|
+
|
11
|
+
master_file = YAML.load_file(paths[:master])
|
12
|
+
slave_file = YAML.load_file(paths[:slave])
|
13
|
+
|
14
|
+
count = diff(master_file[@master[:lang]], slave_file[@slave[:lang]], @master[:lang],
|
15
|
+
@slave[:lang], [@slave[:lang]], [], mode: 'count')
|
16
|
+
|
17
|
+
breakout unless ask_for_permission(@master, count, @slave)
|
18
|
+
|
19
|
+
new_strings_array = diff(master_file[@master[:lang]], slave_file[@slave[:lang]], @master[:lang],
|
20
|
+
@slave[:lang], [@slave[:lang]], [], diff_count: count, mode: 'replace')
|
21
|
+
|
22
|
+
new_strings_array.each do |string|
|
23
|
+
result = string_to_hash(string)
|
24
|
+
merge_hash(result, slave_file)
|
25
|
+
end
|
26
|
+
|
27
|
+
File.open(paths[:slave], 'w') { |file| file.write(slave_file.to_yaml(line_width: -1)) }
|
28
|
+
postpare_paths([paths[:slave]])
|
29
|
+
FileUtils.cp(paths[:slave], @slave[:path])
|
30
|
+
|
31
|
+
reset_tmp_files
|
32
|
+
end
|
33
|
+
end
|
@@ -0,0 +1,33 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.half_automatic
|
5
|
+
tbd
|
6
|
+
@master, @slave, paths = init
|
7
|
+
|
8
|
+
prepare_for_edit(master: @master[:path], slave: @slave[:path])
|
9
|
+
prepare_paths([paths[:master], paths[:slave]])
|
10
|
+
|
11
|
+
master_file = YAML.load_file(paths[:master])
|
12
|
+
slave_file = YAML.load_file(paths[:slave])
|
13
|
+
|
14
|
+
count = diff(master_file[@master[:lang]], slave_file[@slave[:lang]], @master[:lang],
|
15
|
+
@slave[:lang], [@slave[:lang]], [], mode: 'count')
|
16
|
+
|
17
|
+
breakout unless ask_for_permission(@master, count, @slave)
|
18
|
+
|
19
|
+
new_strings_array = diff(master_file[@master[:lang]], slave_file[@slave[:lang]], @master[:lang],
|
20
|
+
@slave[:lang], [@slave[:lang]], [], diff_count: count, mode: 'replace')
|
21
|
+
|
22
|
+
new_strings_array.each do |string|
|
23
|
+
result = string_to_hash(string)
|
24
|
+
merge_hash(result, slave_file)
|
25
|
+
end
|
26
|
+
|
27
|
+
File.open(paths[:slave], 'w') { |file| file.write(slave_file.to_yaml(line_width: -1)) }
|
28
|
+
postpare_paths([paths[:slave]])
|
29
|
+
FileUtils.cp(paths[:slave], @slave[:path])
|
30
|
+
|
31
|
+
reset_tmp_files
|
32
|
+
end
|
33
|
+
end
|
data/lib/tasks/recent.rb
ADDED
@@ -0,0 +1,41 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.recent
|
5
|
+
@master, @slave, paths = init
|
6
|
+
|
7
|
+
FileUtils.cp(@master[:path], paths[:copy])
|
8
|
+
`git checkout "#{@master[:path]}"`
|
9
|
+
|
10
|
+
prepare_for_edit(master: @master[:path], slave: @slave[:path])
|
11
|
+
FileUtils.cp(paths[:copy], @master[:path])
|
12
|
+
|
13
|
+
prepare_paths([paths[:master], paths[:slave], paths[:copy]])
|
14
|
+
|
15
|
+
master_file = YAML.load_file(paths[:master])
|
16
|
+
master_copy_file = YAML.load_file(paths[:copy])
|
17
|
+
slave_file = YAML.load_file(paths[:slave])
|
18
|
+
|
19
|
+
count = diff(master_copy_file[@master[:lang]], master_file[@master[:lang]], @master[:lang],
|
20
|
+
@slave[:lang], [@slave[:lang]], [], mode: 'count', slave_file: slave_file)
|
21
|
+
|
22
|
+
breakout unless ask_for_permission(@master, count, @slave)
|
23
|
+
|
24
|
+
new_strings_array = diff(master_copy_file[@master[:lang]], master_file[@master[:lang]], @master[:lang],
|
25
|
+
@slave[:lang], [@slave[:lang]], [], mode: 'replace', diff_counter: 0, diff_count: count,
|
26
|
+
slave_file: slave_file)
|
27
|
+
|
28
|
+
new_strings_array.each do |string|
|
29
|
+
result = string_to_hash(string)
|
30
|
+
merge_hash(result, slave_file)
|
31
|
+
end
|
32
|
+
|
33
|
+
File.open(paths[:slave], 'w') { |file| file.write(slave_file.to_yaml(line_width: -1)) }
|
34
|
+
postpare_paths([paths[:slave]])
|
35
|
+
FileUtils.cp(paths[:slave], @slave[:path])
|
36
|
+
|
37
|
+
reset_tmp_files
|
38
|
+
log 'Done'
|
39
|
+
log
|
40
|
+
end
|
41
|
+
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.sort_file
|
5
|
+
@master, @slave, paths = init(false)
|
6
|
+
|
7
|
+
prepare_for_edit(master: @master[:path], slave: @slave[:path])
|
8
|
+
prepare_paths([paths[:master]])
|
9
|
+
|
10
|
+
master_file = YAML.load_file(paths[:master])
|
11
|
+
|
12
|
+
if ask_for_sorting(@master[:lang], @master[:app_name])
|
13
|
+
master_file = sort_hash(master_file)
|
14
|
+
File.open(paths[:master], 'w') { |file| file.write(master_file.to_yaml(line_width: -1)) }
|
15
|
+
end
|
16
|
+
|
17
|
+
postpare_paths([paths[:master]])
|
18
|
+
|
19
|
+
FileUtils.cp(paths[:master], @master[:path])
|
20
|
+
|
21
|
+
reset_tmp_files
|
22
|
+
log 'Done'
|
23
|
+
log
|
24
|
+
end
|
25
|
+
end
|
@@ -0,0 +1,37 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.split_file
|
5
|
+
@master, @slave, paths = init(false)
|
6
|
+
|
7
|
+
prepare_for_edit(master: @master[:path])
|
8
|
+
prepare_paths([paths[:master]])
|
9
|
+
|
10
|
+
master_file = YAML.load_file(paths[:master])[@master[:lang]]
|
11
|
+
|
12
|
+
count = master_file.select { |_k, v| v.is_a?(Hash) }.size
|
13
|
+
new_hash = {}
|
14
|
+
|
15
|
+
if ask_for_split(@master[:lang], count)
|
16
|
+
master_file.each_pair do |key, value|
|
17
|
+
next unless value.is_a?(Hash)
|
18
|
+
new_hash[@master[:lang]] = { key => value }
|
19
|
+
File.open(paths[:slave], 'w') { |file| file.write(new_hash.to_yaml(line_width: -1)) }
|
20
|
+
postpare_paths([paths[:slave]])
|
21
|
+
root_path = @master[:path].split('/').reverse.drop(1).reverse.join('/')
|
22
|
+
filename = @master[:path].split('/').last
|
23
|
+
FileUtils.cp(paths[:slave], "#{root_path}/#{key}.#{filename}")
|
24
|
+
master_file.delete(key)
|
25
|
+
end
|
26
|
+
|
27
|
+
new_hash[@master[:lang]] = master_file
|
28
|
+
File.open(paths[:master], 'w') { |file| file.write(new_hash.to_yaml(line_width: -1)) }
|
29
|
+
postpare_paths([paths[:master]])
|
30
|
+
FileUtils.cp(paths[:master], @master[:path])
|
31
|
+
end
|
32
|
+
|
33
|
+
reset_tmp_files
|
34
|
+
log 'Done'
|
35
|
+
log
|
36
|
+
end
|
37
|
+
end
|
data/lib/tasks/start.rb
ADDED
@@ -0,0 +1,59 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module LostInTranslation
|
4
|
+
def self.start
|
5
|
+
tasks = [
|
6
|
+
{ text: 'Compare & Translate two I18n-Locale-Files', method: :start_compare },
|
7
|
+
{ text: 'Count Differences of 2 I18n-Locale-Files', method: :count },
|
8
|
+
{ text: 'Sort an I18n-Locale-File Alphabetically & Recursive (ASC)', method: :sort_file },
|
9
|
+
{ text: 'Split an I18n-Locale-File into multiple namespaced Files (1. Level)', method: :split_file },
|
10
|
+
{ text: 'Show a list of available I18n-Locale-Files', method: :list_files },
|
11
|
+
{ text: 'Tell me joke', method: :joke },
|
12
|
+
{ text: 'Quit', method: :quit }
|
13
|
+
]
|
14
|
+
task = ask_for_task(tasks)
|
15
|
+
LostInTranslation.send(task[:method])
|
16
|
+
end
|
17
|
+
|
18
|
+
def self.start_compare
|
19
|
+
tasks = [
|
20
|
+
{ text: 'Compare everything', method: :compare },
|
21
|
+
{ text: 'Just compare the recent changes (since last git commit)', method: :recent },
|
22
|
+
{ text: 'Automatic translation via API', method: :fully_automatic },
|
23
|
+
{ text: 'Translation suggentions from API', method: :half_automatic },
|
24
|
+
{ text: 'Compare and delete the differences', method: :delete_missing },
|
25
|
+
]
|
26
|
+
task = ask_for_task(tasks)
|
27
|
+
LostInTranslation.send(task[:method])
|
28
|
+
end
|
29
|
+
|
30
|
+
def self.joke
|
31
|
+
system 'clear'
|
32
|
+
log
|
33
|
+
log 'A: Knock, Knock!'
|
34
|
+
log 'B: Who is there?'
|
35
|
+
log 'A: Ya.'
|
36
|
+
log 'B: Ya who?'
|
37
|
+
log 'A: Oh, I prefer google.'
|
38
|
+
log
|
39
|
+
end
|
40
|
+
|
41
|
+
def self.quit
|
42
|
+
system 'clear'
|
43
|
+
log
|
44
|
+
log 'Thank you and goodbye!'
|
45
|
+
log
|
46
|
+
end
|
47
|
+
|
48
|
+
def self.list_files
|
49
|
+
system 'clear'
|
50
|
+
|
51
|
+
path = ask_for_locales_path
|
52
|
+
files = Dir["#{path}/**/*.yml"]
|
53
|
+
puts "No Files found in #{path}" if files.empty?
|
54
|
+
files&.sort&.each do |file|
|
55
|
+
puts file.split('/').last
|
56
|
+
end
|
57
|
+
start
|
58
|
+
end
|
59
|
+
end
|
@@ -0,0 +1,46 @@
|
|
1
|
+
require 'lost_in_translation'
|
2
|
+
|
3
|
+
namespace :lit do
|
4
|
+
desc 'Make the locale-yamls equal again(Interactive)'
|
5
|
+
task :compare do
|
6
|
+
system 'clear'
|
7
|
+
LostInTranslation.compare
|
8
|
+
end
|
9
|
+
|
10
|
+
desc 'Just start and get lead through the process'
|
11
|
+
task :start do
|
12
|
+
system 'clear'
|
13
|
+
LostInTranslation.start
|
14
|
+
end
|
15
|
+
|
16
|
+
desc 'Use Translator-Engine as a suggestion for translation'
|
17
|
+
task :half_automatic do
|
18
|
+
LostInTranslation.half_automatic
|
19
|
+
end
|
20
|
+
|
21
|
+
desc 'Use Translator-Engine to automatically translate everything'
|
22
|
+
task :fully_automatic do
|
23
|
+
LostInTranslation.fully_automatic
|
24
|
+
end
|
25
|
+
|
26
|
+
desc 'Split big files in many small ones'
|
27
|
+
task :split_file do
|
28
|
+
LostInTranslation.split_file
|
29
|
+
end
|
30
|
+
|
31
|
+
desc 'Sort Locale-File recursively'
|
32
|
+
task :sort_file do
|
33
|
+
LostInTranslation.sort_file
|
34
|
+
end
|
35
|
+
|
36
|
+
desc 'Cleaning out the Yaml-Closet'
|
37
|
+
task :delete_missing do
|
38
|
+
LostInTranslation.delete_missing
|
39
|
+
end
|
40
|
+
|
41
|
+
desc 'Tranlate just your recently changed Strings(requires git)'
|
42
|
+
task :recent do
|
43
|
+
abort('Please use this feature inside a Rails-Application') unless defined? Rails
|
44
|
+
LostInTranslation.recent
|
45
|
+
end
|
46
|
+
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
lib = File.expand_path('../lib', __FILE__)
|
4
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
5
|
+
|
6
|
+
Gem::Specification.new do |spec|
|
7
|
+
spec.name = 'lost_in_translation'
|
8
|
+
spec.version = '0.2.19'
|
9
|
+
spec.authors = ['Datyv']
|
10
|
+
spec.email = ['yvesgoizet@gmail.com']
|
11
|
+
|
12
|
+
spec.summary = 'A Gem to compare two locale yamls'
|
13
|
+
spec.description = 'Take two locales yamls and merge them together with different values.'
|
14
|
+
spec.homepage = 'https://github.com/Datyv/lost_in_translation'
|
15
|
+
spec.license = 'MIT'
|
16
|
+
|
17
|
+
spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
|
18
|
+
spec.bindir = 'exe'
|
19
|
+
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
20
|
+
spec.require_paths = ['lib', 'tmp', 'spec']
|
21
|
+
|
22
|
+
spec.add_development_dependency 'bundler', '~> 1.11'
|
23
|
+
spec.add_development_dependency 'rake', '~> 10.0'
|
24
|
+
spec.add_development_dependency 'rspec', '~> 3.0'
|
25
|
+
end
|
data/tmp/tmp_master.yml
ADDED
File without changes
|
File without changes
|
data/tmp/tmp_slave.yml
ADDED
File without changes
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: lost_in_translation
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.2.
|
4
|
+
version: 0.2.19
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Datyv
|
@@ -58,7 +58,36 @@ email:
|
|
58
58
|
executables: []
|
59
59
|
extensions: []
|
60
60
|
extra_rdoc_files: []
|
61
|
-
files:
|
61
|
+
files:
|
62
|
+
- ".gitignore"
|
63
|
+
- ".rspec"
|
64
|
+
- CODE_OF_CONDUCT.md
|
65
|
+
- Gemfile
|
66
|
+
- LICENSE.txt
|
67
|
+
- README.md
|
68
|
+
- Rakefile
|
69
|
+
- bin/console
|
70
|
+
- bin/setup
|
71
|
+
- lib/lit/difference.rb
|
72
|
+
- lib/lit/file_functions.rb
|
73
|
+
- lib/lit/hash.rb
|
74
|
+
- lib/lit/railties.rb
|
75
|
+
- lib/lit/user_interface.rb
|
76
|
+
- lib/lost_in_translation.rb
|
77
|
+
- lib/tasks/compare.rb
|
78
|
+
- lib/tasks/count.rb
|
79
|
+
- lib/tasks/delete_missing.rb
|
80
|
+
- lib/tasks/fully_automatic.rb
|
81
|
+
- lib/tasks/half_automatic.rb
|
82
|
+
- lib/tasks/recent.rb
|
83
|
+
- lib/tasks/sort_file.rb
|
84
|
+
- lib/tasks/split_file.rb
|
85
|
+
- lib/tasks/start.rb
|
86
|
+
- lib/tasks/tasks.rake
|
87
|
+
- lost_in_translation.gemspec
|
88
|
+
- tmp/tmp_master.yml
|
89
|
+
- tmp/tmp_master_copy.yml
|
90
|
+
- tmp/tmp_slave.yml
|
62
91
|
homepage: https://github.com/Datyv/lost_in_translation
|
63
92
|
licenses:
|
64
93
|
- MIT
|