2008-09-21 11:21:30 -04:00
#!/usr/bin/env ruby
begin
$:.unshift(File.expand_path(File.dirname(__FILE__) + '/../lib'))
require 'active_support'
rescue IOError
end
require 'open-uri'
require 'tmpdir'
module ActiveSupport
module Multibyte
2010-05-10 09:46:37 -04:00
module Unicode
2008-09-21 11:21:30 -04:00
2010-05-10 09:46:37 -04:00
class UnicodeDatabase
def load; end
2008-09-21 11:21:30 -04:00
end
2010-05-10 09:46:37 -04:00
class DatabaseGenerator
BASE_URI = "http://www.unicode.org/Public/#{UNICODE_VERSION}/ucd/"
SOURCES = {
:codepoints => BASE_URI + 'UnicodeData.txt',
:composition_exclusion => BASE_URI + 'CompositionExclusions.txt',
:grapheme_break_property => BASE_URI + 'auxiliary/GraphemeBreakProperty.txt',
:cp1252 => 'http://unicode.org/Public/MAPPINGS/VENDORS/MICSFT/WINDOWS/CP1252.TXT'
}
def initialize
@ucd = Unicode::UnicodeDatabase.new
default = Codepoint.new
default.combining_class = 0
default.uppercase_mapping = 0
default.lowercase_mapping = 0
@ucd.codepoints = Hash.new(default)
end
def parse_codepoints(line)
codepoint = Codepoint.new
raise "Could not parse input." unless line =~ /^
([0-9A-F]+); # code
([^;]+); # name
([A-Z]+); # general category
([0-9]+); # canonical combining class
([A-Z]+); # bidi class
(<([A-Z]*)>)? # decomposition type
2011-03-03 23:14:18 -05:00
((\ ?[0-9A-F]+)*); # decomposition mapping
2010-05-10 09:46:37 -04:00
([0-9]*); # decimal digit
([0-9]*); # digit
([^;]*); # numeric
([YN]*); # bidi mirrored
([^;]*); # unicode 1.0 name
([^;]*); # iso comment
([0-9A-F]*); # simple uppercase mapping
([0-9A-F]*); # simple lowercase mapping
([0-9A-F]*)$/ix # simple titlecase mapping
codepoint.code = $1.hex
#codepoint.name = $2
#codepoint.category = $3
codepoint.combining_class = Integer($4)
#codepoint.bidi_class = $5
codepoint.decomp_type = $7
codepoint.decomp_mapping = ($8=='') ? nil : $8.split.collect { |element| element.hex }
#codepoint.bidi_mirrored = ($13=='Y') ? true : false
codepoint.uppercase_mapping = ($16=='') ? 0 : $16.hex
codepoint.lowercase_mapping = ($17=='') ? 0 : $17.hex
#codepoint.titlecase_mapping = ($18=='') ? nil : $18.hex
@ucd.codepoints[codepoint.code] = codepoint
end
def parse_grapheme_break_property(line)
2010-10-15 10:31:00 -04:00
if line =~ /^([0-9A-F.]+)\s*;\s*([\w]+)\s*#/
2010-05-10 09:46:37 -04:00
type = $2.downcase.intern
@ucd.boundary[type] ||= []
if $1.include? '..'
parts = $1.split '..'
@ucd.boundary[type] << (parts[0].hex..parts[1].hex)
else
@ucd.boundary[type] << $1.hex
end
2008-09-21 11:21:30 -04:00
end
end
2010-05-10 09:46:37 -04:00
def parse_composition_exclusion(line)
if line =~ /^([0-9A-F]+)/i
@ucd.composition_exclusion << $1.hex
end
2008-09-21 11:21:30 -04:00
end
2010-05-10 09:46:37 -04:00
def parse_cp1252(line)
if line =~ /^([0-9A-Fx]+)\s([0-9A-Fx]+)/i
@ucd.cp1252[$1.hex] = $2.hex
end
2008-09-21 11:21:30 -04:00
end
2010-05-10 09:46:37 -04:00
def create_composition_map
@ucd.codepoints.each do |_, cp|
if !cp.nil? and cp.combining_class == 0 and cp.decomp_type.nil? and !cp.decomp_mapping.nil? and cp.decomp_mapping.length == 2 and @ucd.codepoints[cp.decomp_mapping[0]].combining_class == 0 and !@ucd.composition_exclusion.include?(cp.code)
@ucd.composition_map[cp.decomp_mapping[0]] ||= {}
@ucd.composition_map[cp.decomp_mapping[0]][cp.decomp_mapping[1]] = cp.code
end
2008-09-21 11:21:30 -04:00
end
end
2010-05-10 09:46:37 -04:00
def normalize_boundary_map
@ucd.boundary.each do |k,v|
2011-04-11 01:35:20 -04:00
if [:lf, :cr].include? k
2010-05-10 09:46:37 -04:00
@ucd.boundary[k] = v[0]
end
2008-09-21 11:21:30 -04:00
end
end
2010-05-10 09:46:37 -04:00
def parse
SOURCES.each do |type, url|
filename = File.join(Dir.tmpdir, "#{url.split('/').last}")
unless File.exist?(filename)
$stderr.puts "Downloading #{url.split('/').last}"
File.open(filename, 'wb') do |target|
open(url) do |source|
source.each_line { |line| target.write line }
end
2008-09-21 11:21:30 -04:00
end
end
2010-05-10 09:46:37 -04:00
File.open(filename) do |file|
file.each_line { |line| send "parse_#{type}".intern, line }
end
2008-09-21 11:21:30 -04:00
end
2010-05-10 09:46:37 -04:00
create_composition_map
normalize_boundary_map
2008-09-21 11:21:30 -04:00
end
2010-05-10 09:46:37 -04:00
def dump_to(filename)
File.open(filename, 'wb') do |f|
f.write Marshal.dump([@ucd.codepoints, @ucd.composition_exclusion, @ucd.composition_map, @ucd.boundary, @ucd.cp1252])
end
2008-09-21 11:21:30 -04:00
end
end
end
end
end
if __FILE__ == $0
2010-05-10 09:46:37 -04:00
filename = ActiveSupport::Multibyte::Unicode::UnicodeDatabase.filename
generator = ActiveSupport::Multibyte::Unicode::DatabaseGenerator.new
2008-09-21 11:21:30 -04:00
generator.parse
print "Writing to: #{filename}"
generator.dump_to filename
puts " (#{File.size(filename)} bytes)"
end