Before adding workers

This commit is contained in:
Jason Jordan
2026-03-20 10:46:53 -04:00
parent 3300819ed5
commit a43c8bf6b5
70 changed files with 533 additions and 457 deletions
@@ -0,0 +1,76 @@
module BenefitsWordDocService
class MapEmployerInformation
def initialize(employer, word_doc_section)
@employer = employer
@card_setup = @employer.id_card_setup
@word_doc_section = word_doc_section
end
def call
search_fields = mapping_hash.keys.to_set
@word_doc_section.each do |line|
matching_field = search_fields.detect { |field| line.include?(field) }
if matching_field
field_mapping = mapping_hash[matching_field]
field_regex = field_mapping[:doc_to_employer_regex]
field_value = line.match(field_regex)[1].strip
employer_field = field_mapping[:employer_field]
if field_mapping[:validation].present?
validation_type = field_mapping[:validation]
if send("is_#{validation_type}?".to_sym, field_value)
@employer[employer_field] = field_value
end
else
@employer[employer_field] = field_value
end
end
end
@card_setup.print_name = @employer.name
@employer.name = @employer.name.titleize
@card_setup.rx_group_number = @employer.group_number
@employer
end
private
def mapping_hash
{
'Employer Name' => {
employer_field: :name,
doc_field: 'Employer Name',
doc_field_desc: 'Follows pattern - Employer Name:New Employer',
doc_to_employer_regex: /.*:(.*)/,
regex_desc: 'Grabs everything after colon'
},
'Group Number' => {
employer_field: :group_number,
doc_field: 'Group Number',
doc_field_desc: 'Follows pattern - Group Number:099999',
doc_to_employer_regex: /.*:(.*)/,
regex_desc: 'Grabs everything after colon',
validation: 'number'
},
'Group Effective Date' => {
employer_field: :effective_date,
doc_field: 'Group Effective Date',
doc_field_desc: 'Follows pattern - Group Effective Date:12/1/2025',
doc_to_employer_regex: /.*:(.*)/,
regex_desc: 'Grabs everything after colon',
validation: 'date'
}
}
end
def is_number?(string)
true if Float(string) rescue false
end
def is_date?(string)
true if Date.parse rescue false
end
end
end
@@ -0,0 +1,78 @@
module BenefitsWordDocService
class MapEmployerLogo
def initialize(employer, word_doc)
@employer = employer
@card_setup = @employer.id_card_setup
@word_doc = word_doc
end
def call
Zip::File.open(@word_doc) do |zip_file|
media_files = zip_file.select { |entry| entry.name.start_with?('word/media/') && !entry.directory? }
if media_files.length > 1
logo = media_files.last
file_extension = File.extname(logo.name)
meme_type = Marcel::MimeType.for(logo.get_input_stream)
image_binary = logo.get_input_stream.read
# image_binary = File.binread(logo.get_input_stream.read)
filename = @employer.name_to_logo_filename(file_extension)
logo = IdCard::EmployerLogo.find_or_create_by(filename: filename) do |clf|
clf.image_data = image_binary
clf.content_type = meme_type
end
# new_logo = CardLogoFile.create!(
# filename: filename,
# image_data: image_binary,
# content_type: meme_type,
# logo_type: "employer"
# )
# image_io = StringIO.new(image_binary)
# width, height = FastImage.size(image_io)
# image_ratio = width.to_f / height
# if (0.8..1.2).cover?(image_ratio)
# @employer.single_card_template = "FairosRxIDCard-Half"
# else
# @employer.single_card_template = "FairosRxIDCard"
# end
@card_setup.employer_logo = logo
end
end
@employer
end
# def call
# extracted_images = []
# Zip::File.open(@word_doc) do |zip_file|
# zip_file.each do |entry|
# if entry.name.start_with?('word/media/') && !entry.directory?
# file_extension = File.extname(entry.name)
# image_data = entry.get_input_stream.read
# extracted_images << { file_extension: file_extension, data: image_data }
# end
# end
# end
# if extracted_images.length > 1
# logo = extracted_images.last
# filename = @employer.employer_name_to_logo_filename(logo[:file_extension])
# employer_logo_binary = logo[:data]
# new_logo = @employer.card_logo_files.create(
# filename: filename,
# image: employer_logo_binary,
# logo_type: "employer"
# )
# @employer.employer_logo = new_logo.filename
# # same file logic
# end
# @employer.save
# end
end
end
@@ -0,0 +1,44 @@
module BenefitsWordDocService
class MapNetworkInformation
def initialize(employer, word_doc_section)
@employer = employer
@card_setup = @employer.id_card_setup
@word_doc_section = word_doc_section
end
def call
# network = @word_doc_section.each do |line|
# if line.match?(/cigna/i)
# return "Cigna"
# elsif line.match?(/medcost/i)
# return "Medcost"
# end
# end
network = @word_doc_section.find do |line|
if line.match?(/cigna/i)
break "Cigna"
elsif line.match?(/medcost/i)
break "MedCost"
end
end
# yellow_fruit_names_filtered = @word_doc_section.filter_map do |line|
# item[:name] if item[:color] == 'yellow'
# end
if network
@card_setup.network_provider = network
logo_name = "#{network}Logo.png"
@card_setup.network_logo = IdCard::NetworkLogo.find_by(filename: logo_name)
@card_setup.provider_section = IdCard::ProviderSection.find_by(title: network)
@card_setup.rx_section = IdCard::RxSection.find_by(title: "FairosRx")
end
@employer
end
end
end
@@ -0,0 +1,278 @@
module BenefitsWordDocService
class MapPlansInformation
def initialize(employer, word_doc_section)
@employer = employer
@card_setup = @employer.id_card_setup
@word_doc_section = word_doc_section
end
def call
search_fields = mapping_hash.keys.to_set
plans_indexes = @word_doc_section.each_index.select { |index| @word_doc_section[index].match?(/\d*\.?\d+k/i) }
plans_indexes.each do |plan_index|
new_plan = @card_setup.plans.build(title: @word_doc_section[plan_index])
plan_lines = @word_doc_section.slice(plan_index + 1, 14)
plan_lines.each_with_index do |line, i|
field_mapping = mapping_array[i]
if line.match(/(?<=:).+/)
field_value = line.match(/(?<=:).+/)[0].strip
else
field_regex = field_mapping[:doc_to_employer_regex]
if line.match(field_regex)
field_value = line.match(field_regex)[0].strip
elsif line.match(default_benefit_regex(field_mapping[:employer_benefit_desc_field]))
field_value = line.match(default_benefit_regex(field_mapping[:employer_benefit_desc_field]))[0].strip
else
field_value = line
end
end
employer_benefit_desc_field = field_mapping[:employer_benefit_desc_field]
new_benefit = new_plan.plan_benefits[i]
new_benefit.benefit = field_value
# matching_field = search_fields.detect { |field| line.include?(field) }
# if matching_field
# field_mapping = mapping_hash[matching_field]
# field_regex = field_mapping[:doc_to_employer_regex]
# if line.match(field_regex)
# field_value = line.match(field_regex)[0].strip
# elsif line.match(default_benefit_regex(field_mapping[:employer_benefit_desc_field]))
# field_value = line.match(default_benefit_regex(field_mapping[:employer_benefit_desc_field]))[0].strip
# else
# field_value = ""
# end
# employer_benefit_desc_field = field_mapping[:employer_benefit_desc_field]
# new_benefit = new_plan.plan_benefits.find_by(benefit_desc: employer_benefit_desc_field)
# new_benefit.benefit = field_value
# new_benefit.save
# end
end
end
@employer
end
private
def mapping_array
[
{
employer_benefit_sequence: 1,
employer_benefit_desc_field: 'Primary Visit',
doc_field_desc: 'Follows pattern - Physician Visit$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 2,
employer_benefit_desc_field: 'Specialist Visit',
doc_field_desc: 'Follows pattern - Specialist Visit$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 3,
employer_benefit_desc_field: 'Urgent Care',
doc_field_desc: 'Follows pattern - Urgent Care$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 4,
employer_benefit_desc_field: 'INNInd Ded',
doc_field_desc: 'Follows pattern - Individual Deductible (in network )$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 5,
employer_benefit_desc_field: 'INNFamily Ded',
doc_field_desc: 'Follows pattern - Family Deductible(in network )$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 6,
employer_benefit_desc_field: 'OONInd Ded',
doc_field_desc: 'Follows pattern - Individual Deductible (out of network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 7,
employer_benefit_desc_field: 'OONFamily Ded',
doc_field_desc: 'Follows pattern - Family Deductible (out of network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 8,
employer_benefit_desc_field: 'Co-Insurance',
doc_field_desc: 'Follows pattern - Co-Insurance70%/30%',
doc_to_employer_regex: /(?<=Co-Insurance).*/,
regex_desc: 'Grabs everything after field name'
},
{
employer_benefit_sequence: 9,
employer_benefit_desc_field: 'INNInd OOP',
doc_field_desc: 'Follows pattern - Out-of-Pocket(in network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 10,
employer_benefit_desc_field: 'INNFamily OOP',
doc_field_desc: 'Follows pattern - Out-of-Pocket Family(in network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 11,
employer_benefit_desc_field: 'OONInd OOP',
doc_field_desc: 'Follows pattern - Out-of-Pocket(out of network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 12,
employer_benefit_desc_field: 'OONFamily OOP',
doc_field_desc: 'Follows pattern - Out-of-Pocket Family (out of network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
{
employer_benefit_sequence: 13,
employer_benefit_desc_field: 'Emergency Room',
doc_field_desc: 'Follows pattern - Emergency RoomXxxxx',
doc_to_employer_regex: /(?<=Emergency Room).*/,
regex_desc: 'Grabs everything after field name'
},
{
employer_benefit_sequence: 14,
employer_benefit_desc_field: 'Preventive Care',
doc_field_desc: 'Follows pattern - Preventive Care100%',
doc_to_employer_regex: /(?<=Preventive Care).*/,
regex_desc: 'Grabs everything after field name'
}
]
end
def mapping_hash
{
'Physician Visit' => {
employer_benefit_sequence: 1,
employer_benefit_desc_field: 'Primary Visit',
doc_field_desc: 'Follows pattern - Physician Visit$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Specialist Visit' => {
employer_benefit_sequence: 2,
employer_benefit_desc_field: 'Specialist Visit',
doc_field_desc: 'Follows pattern - Specialist Visit$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Urgent Care' => {
employer_benefit_sequence: 3,
employer_benefit_desc_field: 'Urgent Care',
doc_field_desc: 'Follows pattern - Urgent Care$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Individual Deductible' => {
employer_benefit_sequence: 4,
employer_benefit_desc_field: 'INNInd Ded',
doc_field_desc: 'Follows pattern - Individual Deductible (in network )$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Family Deductible' => {
employer_benefit_sequence: 5,
employer_benefit_desc_field: 'INNFamily Ded',
doc_field_desc: 'Follows pattern - Family Deductible(in network )$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Individual Deductible' => {
employer_benefit_sequence: 6,
employer_benefit_desc_field: 'OONInd Ded',
doc_field_desc: 'Follows pattern - Individual Deductible (out of network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Family Deductible' => {
employer_benefit_sequence: 7,
employer_benefit_desc_field: 'OONFamily Ded',
doc_field_desc: 'Follows pattern - Family Deductible (out of network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Co-Insurance' => {
employer_benefit_sequence: 8,
employer_benefit_desc_field: 'Co-Insurance',
doc_field_desc: 'Follows pattern - Co-Insurance70%/30%',
doc_to_employer_regex: /(?<=Co-Insurance).*/,
regex_desc: 'Grabs everything after field name'
},
'Out-of-Pocket' => {
employer_benefit_sequence: 9,
employer_benefit_desc_field: 'INNInd OOP',
doc_field_desc: 'Follows pattern - Out-of-Pocket(in network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Out-of-Pocket Family' => {
employer_benefit_sequence: 10,
employer_benefit_desc_field: 'INNFamily OOP',
doc_field_desc: 'Follows pattern - Out-of-Pocket Family(in network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Out-of-Pocket' => {
employer_benefit_sequence: 11,
employer_benefit_desc_field: 'OONInd OOP',
doc_field_desc: 'Follows pattern - Out-of-Pocket(out of network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Out-of-Pocket Family' => {
employer_benefit_sequence: 12,
employer_benefit_desc_field: 'OONFamily OOP',
doc_field_desc: 'Follows pattern - Out-of-Pocket Family (out of network)$x,xxx',
doc_to_employer_regex: /\$.*/,
regex_desc: 'Grabs dollar sign and everything after'
},
'Emergency Room' => {
employer_benefit_sequence: 13,
employer_benefit_desc_field: 'Emergency Room',
doc_field_desc: 'Follows pattern - Emergency RoomXxxxx',
doc_to_employer_regex: /(?<=Emergency Room).*/,
regex_desc: 'Grabs everything after field name'
},
'Preventive Care' => {
employer_benefit_sequence: 14,
employer_benefit_desc_field: 'Preventive Care',
doc_field_desc: 'Follows pattern - Preventive Care100%',
doc_to_employer_regex: /(?<=Preventive Care).*/,
regex_desc: 'Grabs everything after field name'
},
}
end
def default_benefit_regex(field)
/(?<=#{field}).*/
end
def is_number?(string)
true if Float(string) rescue false
end
def is_date?(string)
true if Date.parse rescue false
end
end
end
@@ -0,0 +1,35 @@
module BenefitsWordDocService
class WordDocProcessor
def initialize(word_doc, employer=nil)
@word_doc = word_doc
if employer
@employer = employer
else
@employer = Employer.new
@employer.build_id_card_setup
end
end
def call
doc = Docx::Document.open(@word_doc)
data_lines = doc.paragraphs.map { |p| p.to_s.squish }.reject!(&:empty?)
employer_information, plans_and_network = data_lines.split("Medical Plan")
plan_information, network_information = plans_and_network.split("Claims Submission")
# employer_information = data_lines.slice(0, start_of_plans_index)
# plan_information = data_lines.slice(start_of_plans_index + 1..)
# network_information = data_lines.slice(start_of_network_index + 1..)
@employer = BenefitsWordDocService::MapEmployerInformation.new(@employer, employer_information).call
@employer = BenefitsWordDocService::MapEmployerLogo.new(@employer, @word_doc).call
@employer = BenefitsWordDocService::MapPlansInformation.new(@employer, plan_information).call
@employer = BenefitsWordDocService::MapNetworkInformation.new(@employer, network_information).call
@employer
end
end
end