1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
|
# frozen_string_literal: true
require 'mail/utilities'
require 'mail/parser_tools'
begin
original_verbose, $VERBOSE = $VERBOSE, nil
%%{
machine content_disposition;
alphtype int;
# Disposition Type
action disp_type_s { disp_type_s = p }
action disp_type_e { content_disposition.disposition_type = chars(data, disp_type_s, p-1).downcase }
# Parameter Attribute
action param_attr_s { param_attr_s = p }
action param_attr_e { param_attr = chars(data, param_attr_s, p-1) }
# Quoted String
action qstr_s { qstr_s = p }
action qstr_e { qstr = chars(data, qstr_s, p-1) }
# Parameter Value
action param_val_s { param_val_s = p }
action param_val_e {
if param_attr.nil?
raise Mail::Field::ParseError.new(Mail::ContentDispositionElement, data, "no attribute for value")
end
# Use quoted string value if one exists, otherwise use parameter value
value = qstr || chars(data, param_val_s, p-1)
content_disposition.parameters << { param_attr => value }
param_attr = nil
qstr = nil
}
# No-op actions
action comment_e { }
action comment_s { }
action phrase_e { }
action phrase_s { }
action main_type_e { }
action main_type_s { }
action sub_type_e { }
action sub_type_s { }
include rfc2183_content_disposition "rfc2183_content_disposition.rl";
main := disposition;
}%%
module Mail::Parsers
module ContentDispositionParser
extend Mail::ParserTools
ContentDispositionStruct = Struct.new(:disposition_type, :parameters, :error)
%%write data noprefix;
def self.parse(data)
data = data.dup.force_encoding(Encoding::ASCII_8BIT) if data.respond_to?(:force_encoding)
content_disposition = ContentDispositionStruct.new('', [])
return content_disposition if Mail::Utilities.blank?(data)
# Parser state
disp_type_s = param_attr_s = param_attr = qstr_s = qstr = param_val_s = nil
# 5.1 Variables Used by Ragel
p = 0
eof = pe = data.length
stack = []
%%write init;
%%write exec;
if p != eof || cs < %%{ write first_final; }%%
raise Mail::Field::IncompleteParseError.new(Mail::ContentDispositionElement, data, p)
end
content_disposition
end
end
end
ensure
$VERBOSE = original_verbose
end
|