Couple of high level thoughts (attempted with minimal caffeination, so take with large conglomeration of NaCL and I may be whiffing on template syntax a bit):
This is almost complex enough you might want to look at using a proper parser (Parse::RecDescent, Marpa::R2) and build some sort of data structure to represent each "interface entry". Alternately maybe at least use "!\n" as your record separator and push the existing parsing logic off into its own sub (which then returns some sort of data structure . . .); doing that might help make sure you're not intermingling blocks and can (slightly more easily) start with a fresh parse state each one.
Also rather than a bunch of hardcoded print statements for something like this I'd use Template Toolkit (see also Template) and write something which takes the data produced by the parsing bits and then passes those to something which expands into the final output lines.
[% MACRO interface_metainfo( if_info ) BEGIN %]
set interfaces ge-1/1/1 unit 23 description "[% if_info.description %]
+"
set interfaces ge-1/1/1 unit 23 encapsulation [% if_info.encapsulation
+ %]
set interfaces ge-1/1/1 unit 23 vlan-tags [% FOR tag IN if_info.vlan_t
+ags.keys %][% tag %] [% if_info.vlan_tags.$tag %][% END %]
## YADDA YADDA YADDA
[% END %]
[%- CLEAR -%]
[% FOR if IN interface_list %]
[% interface_metainfo( if ) %]
[% protocol_block( if ) %]
[% class_of_service_block( if ) %]
[% END %]
The cake is a lie.
The cake is a lie.
The cake is a lie.