2016-02-28 09:49:57 +00:00
|
|
|
# coding: utf-8
|
|
|
|
|
|
|
|
def format_string_for_po str
|
|
|
|
return '"' + str.gsub(/"/, '\"') + '"' unless /\\n./.match(str)
|
|
|
|
|
|
|
|
([ '""' ] + str.split(/(?<=\\n)/).map { |part| '"' + part.gsub(/"/, '\"') + '"' }).join("\n")
|
|
|
|
end
|
|
|
|
|
|
|
|
def unformat_string_for_po str
|
|
|
|
str.gsub(/^"|"$/, '').gsub(/\\"/, '"')
|
|
|
|
end
|
|
|
|
|
2016-11-01 22:19:06 +00:00
|
|
|
def fix_po_msgstr_plurals items
|
|
|
|
return items if (items.size == 0) || !items.first.key?(:comments)
|
|
|
|
|
|
|
|
matches = %r{nplurals=(\d+)}.match(items.first[:comments].join(""))
|
|
|
|
|
|
|
|
return items if !matches
|
|
|
|
|
|
|
|
num_plurals = matches[1].to_i
|
|
|
|
|
|
|
|
return items if num_plurals <= 0
|
|
|
|
|
|
|
|
items.each do |item|
|
|
|
|
next unless item.key?(:msgstr)
|
|
|
|
item[:msgstr] = item[:msgstr][0..num_plurals - 1] if item[:msgstr].size > num_plurals
|
|
|
|
end
|
|
|
|
|
|
|
|
return items
|
|
|
|
end
|
|
|
|
|
2016-02-28 09:49:57 +00:00
|
|
|
def read_po file_name
|
|
|
|
items = [ { comments: [] } ]
|
|
|
|
msgtype = nil
|
|
|
|
line_no = 0
|
2016-07-11 18:47:03 +00:00
|
|
|
started = false
|
2016-02-28 09:49:57 +00:00
|
|
|
|
|
|
|
add_line = lambda do |type, to_add|
|
|
|
|
items.last[type] ||= []
|
|
|
|
items.last[type] += to_add if to_add.is_a?(Array)
|
|
|
|
items.last[type] << to_add if to_add.is_a?(String)
|
|
|
|
end
|
|
|
|
|
|
|
|
IO.readlines(file_name).each do |line|
|
|
|
|
line_no += 1
|
|
|
|
line.chomp!
|
|
|
|
|
2016-07-11 18:47:03 +00:00
|
|
|
if !line.empty? && !started
|
|
|
|
items.last[:line] = line_no
|
|
|
|
started = true
|
|
|
|
end
|
|
|
|
|
2016-02-28 09:49:57 +00:00
|
|
|
if line.empty?
|
|
|
|
items << {} unless items.last.keys.empty?
|
|
|
|
msgtype = nil
|
2016-07-11 18:47:03 +00:00
|
|
|
started = false
|
2016-02-28 09:49:57 +00:00
|
|
|
|
|
|
|
elsif items.size == 1
|
|
|
|
add_line.call :comments, line
|
|
|
|
|
|
|
|
elsif /^#:\s*(.+)/.match(line)
|
|
|
|
add_line.call :sources, $1.split(/\s+/)
|
|
|
|
|
|
|
|
elsif /^#,\s*(.+)/.match(line)
|
|
|
|
add_line.call :flags, $1.split(/,\s*/)
|
|
|
|
|
|
|
|
elsif /^#\./.match(line)
|
|
|
|
add_line.call :instructions, line
|
|
|
|
|
|
|
|
elsif /^#~/.match(line)
|
|
|
|
add_line.call :obsolete, line
|
|
|
|
|
|
|
|
elsif /^#\|/.match(line)
|
2016-02-28 14:56:05 +00:00
|
|
|
add_line.call :suggestions, line
|
2016-02-28 09:49:57 +00:00
|
|
|
|
|
|
|
elsif /^#\s/.match(line)
|
|
|
|
add_line.call :comments, line
|
|
|
|
|
|
|
|
elsif /^ ( msgid(?:_plural)? | msgstr (?: \[ \d+ \])? ) \s* (.+)/x.match(line)
|
|
|
|
type, string = $1, $2
|
|
|
|
msgtype = type.gsub(/\[.*/, '').to_sym
|
|
|
|
|
|
|
|
items.last[msgtype] ||= []
|
|
|
|
items.last[msgtype] << unformat_string_for_po(string)
|
|
|
|
|
|
|
|
elsif /^"/.match(line)
|
|
|
|
fail "read_po: #{file_name}:#{line_no}: string entry without prior msgid/msgstr for »#{line}«" unless msgtype
|
|
|
|
items.last[msgtype].last << unformat_string_for_po(line)
|
|
|
|
|
|
|
|
else
|
|
|
|
fail "read_po: #{file_name}:#{line_no}: unrecognized line type for »#{line}«"
|
|
|
|
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
items.pop if items.last.keys.empty?
|
|
|
|
|
2016-11-01 22:19:06 +00:00
|
|
|
return fix_po_msgstr_plurals items
|
2016-02-28 09:49:57 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
def write_po file_name, items
|
|
|
|
File.open(file_name, "w") do |file|
|
2016-07-11 19:52:43 +00:00
|
|
|
|
2016-02-28 09:49:57 +00:00
|
|
|
items.each do |item|
|
|
|
|
if item[:obsolete]
|
|
|
|
file.puts(item[:obsolete].join("\n"))
|
|
|
|
file.puts
|
|
|
|
next
|
|
|
|
end
|
|
|
|
|
2016-02-28 10:44:58 +00:00
|
|
|
if item[:comments] && !item[:comments].empty?
|
2016-02-28 09:49:57 +00:00
|
|
|
file.puts(item[:comments].join("\n"))
|
|
|
|
end
|
|
|
|
|
2016-02-28 10:44:58 +00:00
|
|
|
if item[:instructions] && !item[:instructions].empty?
|
2016-02-28 09:49:57 +00:00
|
|
|
file.puts(item[:instructions].join("\n"))
|
|
|
|
end
|
|
|
|
|
2016-02-28 10:44:58 +00:00
|
|
|
if item[:sources] && !item[:sources].empty?
|
2016-02-28 09:49:57 +00:00
|
|
|
file.puts(item[:sources].map { |source| "#: #{source}" }.join("\n").gsub(/,$/, ''))
|
|
|
|
end
|
|
|
|
|
2016-02-28 10:44:58 +00:00
|
|
|
if item[:flags] && !item[:flags].empty?
|
2016-02-28 09:49:57 +00:00
|
|
|
file.puts("#, " + item[:flags].join(", "))
|
|
|
|
end
|
|
|
|
|
2016-02-28 14:56:05 +00:00
|
|
|
if item[:suggestions] && !item[:suggestions].empty?
|
|
|
|
file.puts(item[:suggestions].join("\n"))
|
2016-02-28 09:49:57 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
if item[:msgid]
|
|
|
|
file.puts("msgid " + format_string_for_po(item[:msgid].first))
|
|
|
|
end
|
|
|
|
|
|
|
|
if item[:msgid_plural]
|
|
|
|
file.puts("msgid_plural " + format_string_for_po(item[:msgid_plural].first))
|
|
|
|
end
|
|
|
|
|
|
|
|
if item[:msgstr]
|
|
|
|
idx = 0
|
|
|
|
|
|
|
|
item[:msgstr].each do |msgstr|
|
|
|
|
suffix = item[:msgid_plural] ? "[#{idx}]" : ""
|
|
|
|
idx += 1
|
|
|
|
file.puts("msgstr#{suffix} " + format_string_for_po(msgstr))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
file.puts
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def normalize_po file
|
2016-04-10 13:22:19 +00:00
|
|
|
puts_action "NORMALIZE-PO", file
|
2016-02-28 09:49:57 +00:00
|
|
|
write_po file, read_po(file)
|
|
|
|
end
|
2016-02-28 10:44:58 +00:00
|
|
|
|
2016-02-28 15:37:03 +00:00
|
|
|
def replace_po_meta_info orig_metas, transifex_meta, key
|
|
|
|
new_value = /"#{key}: \s+ (.+?) \\n"/x.match(transifex_meta)[1]
|
|
|
|
# puts "looking for #{key} in #{transifex_meta}"
|
|
|
|
# puts " new val #{new_value}"
|
|
|
|
return unless new_value
|
|
|
|
|
|
|
|
orig_metas.each { |meta| meta.gsub!(/"#{key}: \s+ .+? \\n"/x, "\"#{key}: #{new_value}\\n\"") }
|
|
|
|
end
|
|
|
|
|
2016-11-20 16:27:08 +00:00
|
|
|
def merge_po orig_items, updated_items, options = {}
|
2016-05-23 07:31:01 +00:00
|
|
|
translated = Hash[ *updated_items.
|
2016-02-28 10:44:58 +00:00
|
|
|
select { |item| item[:msgid] && item[:msgid].first && !item[:msgid].first.empty? && item[:msgstr] && !item[:msgstr].empty? && !item[:msgstr].first.empty? }.
|
|
|
|
map { |item| [ item[:msgid].first, item ] }.
|
|
|
|
flatten(1)
|
|
|
|
]
|
|
|
|
|
2016-02-28 15:37:03 +00:00
|
|
|
update_meta_info = false
|
|
|
|
|
2016-02-28 10:44:58 +00:00
|
|
|
orig_items.each do |orig_item|
|
|
|
|
next if !orig_item[:msgid] || orig_item[:msgid].empty? || orig_item[:msgid].first.empty?
|
|
|
|
|
2016-05-23 07:31:01 +00:00
|
|
|
updated_item = translated[ orig_item[:msgid].first ]
|
2016-02-28 10:44:58 +00:00
|
|
|
|
2016-05-23 07:31:01 +00:00
|
|
|
next if !updated_item
|
2016-04-02 08:54:53 +00:00
|
|
|
|
2016-05-23 07:31:01 +00:00
|
|
|
next if (orig_item[:msgstr] == updated_item[:msgstr]) && !(orig_item[:flags] || []).include?("fuzzy")
|
2016-02-28 10:44:58 +00:00
|
|
|
|
|
|
|
# puts "UPDATE of msgid " + orig_item[:msgid].first
|
|
|
|
# puts " old " + orig_item[:msgstr].first
|
2016-05-23 07:31:01 +00:00
|
|
|
# puts " new " + updated_item[:msgstr].first
|
2016-02-28 10:44:58 +00:00
|
|
|
|
2016-02-28 15:37:03 +00:00
|
|
|
update_meta_info = true
|
2016-05-23 07:31:01 +00:00
|
|
|
orig_item[:msgstr] = updated_item[:msgstr]
|
2016-02-28 10:44:58 +00:00
|
|
|
|
2016-07-23 12:20:05 +00:00
|
|
|
next if (updated_item[:flags] || []).include?("fuzzy")
|
|
|
|
|
2016-02-28 14:56:05 +00:00
|
|
|
orig_item[:flags].reject! { |flag| flag == "fuzzy" } if orig_item[:flags]
|
|
|
|
orig_item.delete(:suggestions)
|
2016-02-28 10:44:58 +00:00
|
|
|
end
|
|
|
|
|
2016-02-28 15:37:03 +00:00
|
|
|
# update_meta_info = true
|
|
|
|
|
2016-11-20 16:27:08 +00:00
|
|
|
return orig_items unless update_meta_info
|
2016-02-28 15:37:03 +00:00
|
|
|
|
2016-11-20 16:27:08 +00:00
|
|
|
orig_meta = orig_items.first[:comments]
|
|
|
|
updated_meta = updated_items.first[:comments].join("")
|
|
|
|
headers_to_update = %w{PO-Revision-Date Last-Translator Language-Team Plural-Forms}
|
|
|
|
headers_to_update += options[:headers_to_update] || []
|
|
|
|
|
|
|
|
headers_to_update.each { |key| replace_po_meta_info orig_meta, updated_meta, key }
|
2016-02-28 15:37:03 +00:00
|
|
|
|
2016-02-28 10:44:58 +00:00
|
|
|
orig_items
|
|
|
|
end
|
|
|
|
|
|
|
|
def transifex_pull_and_merge resource, language
|
|
|
|
po_file = resource == "programs" ? "po/#{language}.po" : "doc/man/po4a/po/#{language}.po"
|
|
|
|
|
|
|
|
runq_git po_file, "checkout HEAD -- #{po_file}"
|
|
|
|
|
|
|
|
orig_items = read_po(po_file)
|
|
|
|
|
2016-03-26 11:34:52 +00:00
|
|
|
runq "tx pull", po_file, "tx pull -f -r mkvtoolnix.#{resource} -l #{language} > /dev/null"
|
2016-02-28 10:44:58 +00:00
|
|
|
|
2016-03-28 15:38:55 +00:00
|
|
|
puts_qaction "merge", po_file
|
2016-02-28 10:44:58 +00:00
|
|
|
|
|
|
|
transifex_items = read_po(po_file)
|
2016-05-23 07:31:01 +00:00
|
|
|
merged_items = merge_po orig_items, transifex_items
|
2016-11-01 22:19:06 +00:00
|
|
|
fixed_items = fix_po_msgstr_plurals merged_items
|
2016-02-28 10:44:58 +00:00
|
|
|
|
2016-11-01 22:19:06 +00:00
|
|
|
write_po po_file, fixed_items
|
2016-02-28 10:44:58 +00:00
|
|
|
end
|
2016-03-26 11:15:11 +00:00
|
|
|
|
|
|
|
def transifex_remove_fuzzy_and_push resource, language
|
|
|
|
po_file = resource == "programs" ? "po/#{language}.po" : "doc/man/po4a/po/#{language}.po"
|
|
|
|
po_file_no_fuzzy = Tempfile.new("mkvtoolnix-rake-po-no-fuzzy")
|
|
|
|
|
|
|
|
runq_git po_file, "checkout HEAD -- #{po_file}"
|
|
|
|
|
2016-03-26 11:34:52 +00:00
|
|
|
runq "msgattrib", po_file, "msgattrib --no-fuzzy --output=#{po_file_no_fuzzy.path} #{po_file}"
|
2016-03-26 11:15:11 +00:00
|
|
|
|
|
|
|
IO.write(po_file, IO.read(po_file_no_fuzzy))
|
|
|
|
|
|
|
|
normalize_po po_file
|
|
|
|
|
2016-03-26 11:34:52 +00:00
|
|
|
runq "tx push", po_file, "tx push -t -f --no-interactive -r mkvtoolnix.#{resource} -l #{language} > /dev/null"
|
2016-03-26 11:15:11 +00:00
|
|
|
|
|
|
|
runq_git po_file, "checkout HEAD -- #{po_file}"
|
|
|
|
end
|