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
|
# frozen_string_literal: true
require 'mail/utilities'
require 'mail/parser_tools'
begin
original_verbose, $VERBOSE = $VERBOSE, nil
%%{
# RFC 2557 Content-Location
# https://tools.ietf.org/html/rfc2557#section-4.1
machine content_location;
alphtype int;
# Quoted String
action qstr_s { qstr_s = p }
action qstr_e { content_location.location = chars(data, qstr_s, p-1) }
# Token String
action token_string_s { token_string_s = p }
action token_string_e { content_location.location = chars(data, token_string_s, p-1) }
# 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 { }
action param_attr_e { }
action param_attr_s { }
action param_val_e { }
action param_val_s { }
include rfc2045_content_type "rfc2045_content_type.rl";
location = quoted_string | ((token | 0x3d)+ >token_string_s %token_string_e);
main := CFWS? location CFWS?;
}%%
module Mail::Parsers
module ContentLocationParser
extend Mail::ParserTools
ContentLocationStruct = Struct.new(:location, :error)
%%write data noprefix;
def self.parse(data)
data = data.dup.force_encoding(Encoding::ASCII_8BIT) if data.respond_to?(:force_encoding)
content_location = ContentLocationStruct.new(nil)
return content_location 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::ContentLocationElement, data, p)
end
content_location
end
end
end
ensure
$VERBOSE = original_verbose
end
|