diff --git a/data/templates/frr/ospf6d.frr.j2 b/data/templates/frr/ospf6d.frr.j2
index 84394ed1a..b0b5663dd 100644
--- a/data/templates/frr/ospf6d.frr.j2
+++ b/data/templates/frr/ospf6d.frr.j2
@@ -1,95 +1,116 @@
 !
 {% if interface is vyos_defined %}
 {%     for iface, iface_config in interface.items() %}
 interface {{ iface }}
 {%         if iface_config.area is vyos_defined %}
  ipv6 ospf6 area {{ iface_config.area }}
 {%         endif %}
 {%         if iface_config.cost is vyos_defined %}
  ipv6 ospf6 cost {{ iface_config.cost }}
 {%         endif %}
 {%         if iface_config.priority is vyos_defined %}
  ipv6 ospf6 priority {{ iface_config.priority }}
 {%         endif %}
 {%         if iface_config.hello_interval is vyos_defined %}
  ipv6 ospf6 hello-interval {{ iface_config.hello_interval }}
 {%         endif %}
 {%         if iface_config.retransmit_interval is vyos_defined %}
  ipv6 ospf6 retransmit-interval {{ iface_config.retransmit_interval }}
 {%         endif %}
 {%         if iface_config.transmit_delay is vyos_defined %}
  ipv6 ospf6 transmit-delay {{ iface_config.transmit_delay }}
 {%         endif %}
 {%         if iface_config.dead_interval is vyos_defined %}
  ipv6 ospf6 dead-interval {{ iface_config.dead_interval }}
 {%         endif %}
 {%         if iface_config.bfd is vyos_defined %}
  ipv6 ospf6 bfd
 {%         endif %}
 {%         if iface_config.bfd.profile is vyos_defined %}
  ipv6 ospf6 bfd profile {{ iface_config.bfd.profile }}
 {%         endif %}
 {%         if iface_config.mtu_ignore is vyos_defined %}
  ipv6 ospf6 mtu-ignore
 {%         endif %}
 {%         if iface_config.ifmtu is vyos_defined %}
  ipv6 ospf6 ifmtu {{ iface_config.ifmtu }}
 {%         endif %}
 {%         if iface_config.network is vyos_defined %}
  ipv6 ospf6 network {{ iface_config.network }}
 {%         endif %}
 {%         if iface_config.instance_id is vyos_defined %}
  ipv6 ospf6 instance-id {{ iface_config.instance_id }}
 {%         endif %}
 {%         if iface_config.passive is vyos_defined %}
  ipv6 ospf6 passive
 {%         endif %}
 exit
 !
 {%     endfor %}
 {% endif %}
 !
 router ospf6 {{ 'vrf ' ~ vrf if vrf is vyos_defined }}
 {% if area is vyos_defined %}
 {%     for area_id, area_config in area.items() %}
 {%         if area_config.area_type is vyos_defined %}
 {%             for type, type_config in area_config.area_type.items() %}
  area {{ area_id }} {{ type }} {{ 'default-information-originate' if type_config.default_information_originate is vyos_defined }} {{ 'no-summary' if type_config.no_summary is vyos_defined }}
 {%             endfor %}
 {%         endif %}
 {%         if area_config.range is vyos_defined %}
 {%             for prefix, prefix_config in area_config.range.items() %}
  area {{ area_id }} range {{ prefix }} {{ 'advertise' if prefix_config.advertise is vyos_defined }} {{ 'not-advertise' if prefix_config.not_advertise is vyos_defined }}
 {%             endfor %}
 {%         endif %}
 {%         if area_config.export_list is vyos_defined %}
  area {{ area_id }} export-list {{ area_config.export_list }}
 {%         endif %}
 {%         if area_config.import_list is vyos_defined %}
  area {{ area_id }} import-list {{ area_config.import_list }}
 {%         endif %}
 {%     endfor %}
 {% endif %}
  auto-cost reference-bandwidth {{ auto_cost.reference_bandwidth }}
 {% if default_information.originate is vyos_defined %}
  default-information originate {{ 'always' if default_information.originate.always is vyos_defined }} {{ 'metric ' ~ default_information.originate.metric if default_information.originate.metric is vyos_defined }} {{ 'metric-type ' ~ default_information.originate.metric_type if default_information.originate.metric_type is vyos_defined }} {{ 'route-map ' ~ default_information.originate.route_map if default_information.originate.route_map is vyos_defined }}
 {% endif %}
 {% if distance.global is vyos_defined %}
  distance {{ distance.global }}
 {% endif %}
 {% if distance.ospfv3 is vyos_defined %}
  distance ospf6 {{ 'intra-area ' ~ distance.ospfv3.intra_area if distance.ospfv3.intra_area is vyos_defined }} {{ 'inter-area ' ~ distance.ospfv3.inter_area if distance.ospfv3.inter_area is vyos_defined }} {{ 'external ' ~ distance.ospfv3.external if distance.ospfv3.external is vyos_defined }}
 {% endif %}
+{% if graceful_restart is vyos_defined %}
+{%     if graceful_restart.grace_period is vyos_defined %}
+ graceful-restart grace-period {{ graceful_restart.grace_period }}
+{%     endif %}
+{%     if graceful_restart.helper.enable.router_id is vyos_defined %}
+{%         for router_id in graceful_restart.helper.enable.router_id %}
+ graceful-restart helper enable {{ router_id }}
+{%         endfor %}
+{%     elif graceful_restart.helper.enable is vyos_defined %}
+ graceful-restart helper enable
+{%     endif %}
+{%     if graceful_restart.helper.planned_only is vyos_defined %}
+ graceful-restart helper planned-only
+{%     endif %}
+{%     if graceful_restart.helper.lsa_check_disable is vyos_defined %}
+ graceful-restart helper lsa-check-disable
+{%     endif %}
+{%     if graceful_restart.helper.supported_grace_time is vyos_defined %}
+ graceful-restart helper supported-grace-time {{ graceful_restart.helper.supported_grace_time }}
+{%     endif %}
+{% endif %}
 {% if log_adjacency_changes is vyos_defined %}
  log-adjacency-changes {{ "detail" if log_adjacency_changes.detail is vyos_defined }}
 {% endif %}
 {% if parameters.router_id is vyos_defined %}
  ospf6 router-id {{ parameters.router_id }}
 {% endif %}
 {% if redistribute is vyos_defined %}
 {%     for protocol, options in redistribute.items() %}
  redistribute {{ protocol }} {{ 'route-map ' ~ options.route_map if options.route_map is vyos_defined }}
 {%     endfor %}
 {% endif %}
 exit
 !
diff --git a/data/templates/frr/ospfd.frr.j2 b/data/templates/frr/ospfd.frr.j2
index 1ee8d8752..040628e82 100644
--- a/data/templates/frr/ospfd.frr.j2
+++ b/data/templates/frr/ospfd.frr.j2
@@ -1,238 +1,262 @@
 !
 {% if interface is vyos_defined %}
 {%     for iface, iface_config in interface.items() %}
 interface {{ iface }}
 {%         if iface_config.authentication.plaintext_password is vyos_defined %}
  ip ospf authentication-key {{ iface_config.authentication.plaintext_password }}
 {%         elif iface_config.authentication.md5 is vyos_defined %}
  ip ospf authentication message-digest
 {%             if iface_config.authentication.md5.key_id is vyos_defined %}
 {%                 for key, key_config in iface_config.authentication.md5.key_id.items() %}
  ip ospf message-digest-key {{ key }} md5 {{ key_config.md5_key }}
 {%                 endfor %}
 {%             endif %}
 {%         endif %}
 {%         if iface_config.area is vyos_defined %}
  ip ospf area {{ iface_config.area }}
 {%         endif %}
 {%         if iface_config.bandwidth is vyos_defined %}
  bandwidth {{ iface_config.bandwidth }}
 {%         endif %}
 {%         if iface_config.cost is vyos_defined %}
  ip ospf cost {{ iface_config.cost }}
 {%         endif %}
 {%         if iface_config.priority is vyos_defined %}
  ip ospf priority {{ iface_config.priority }}
 {%         endif %}
 {%         if iface_config.hello_interval is vyos_defined %}
  ip ospf hello-interval {{ iface_config.hello_interval }}
 {%         endif %}
 {%         if iface_config.retransmit_interval is vyos_defined %}
  ip ospf retransmit-interval {{ iface_config.retransmit_interval }}
 {%         endif %}
 {%         if iface_config.transmit_delay is vyos_defined %}
  ip ospf transmit-delay {{ iface_config.transmit_delay }}
 {%         endif %}
 {%         if iface_config.dead_interval is vyos_defined %}
  ip ospf dead-interval {{ iface_config.dead_interval }}
 {%         elif iface_config.hello_multiplier is vyos_defined %}
  ip ospf dead-interval minimal hello-multiplier {{ iface_config.hello_multiplier }}
 {%         endif %}
 {%         if iface_config.bfd is vyos_defined %}
  ip ospf bfd
 {%         endif %}
 {%         if iface_config.bfd.profile is vyos_defined %}
  ip ospf bfd profile {{ iface_config.bfd.profile }}
 {%         endif %}
 {%         if iface_config.ldp_sync.disable is vyos_defined %}
  no ip ospf mpls ldp-sync
 {%         elif iface_config.ldp_sync.holddown is vyos_defined %}
  ip ospf mpls ldp-sync
  ip ospf mpls ldp-sync holddown {{ iface_config.ldp_sync.holddown }}
 {%         endif %}
 {%         if iface_config.mtu_ignore is vyos_defined %}
  ip ospf mtu-ignore
 {%         endif %}
 {%         if iface_config.network is vyos_defined %}
  ip ospf network {{ iface_config.network }}
 {%         endif %}
 {%         if iface_config.passive is vyos_defined %}
  {{ 'no ' if iface_config.passive.disable is vyos_defined }}ip ospf passive
 {%         endif %}
 exit
 !
 {%     endfor %}
 {% endif %}
 !
 router ospf {{ 'vrf ' ~ vrf if vrf is vyos_defined }}
 {% if access_list is vyos_defined %}
 {%     for acl, acl_config in access_list.items() %}
 {%         for protocol in acl_config.export if acl_config.export is vyos_defined %}
  distribute-list {{ acl }} out {{ protocol }}
 {%         endfor %}
 {%     endfor %}
 {% endif %}
 {% if aggregation.timer is vyos_defined %}
  aggregation timer {{ aggregation.timer }}
 {% endif %}
 {% if area is vyos_defined %}
 {%     for area_id, area_config in area.items() %}
 {%         if area_config.area_type is vyos_defined %}
 {%             for type, type_config in area_config.area_type.items() if type != 'normal' %}
  area {{ area_id }} {{ type }} {{ 'no-summary' if type_config.no_summary is vyos_defined }}
 {%                 if type_config.default_cost is vyos_defined %}
  area {{ area_id }} default-cost {{ type_config.default_cost }}
 {%                 endif %}
 {%             endfor %}
 {%         endif %}
 {%         if area_config.authentication is vyos_defined %}
  area {{ area_id }} authentication {{ 'message-digest' if area_config.authentication is vyos_defined('md5') }}
 {%         endif %}
 {%         for network in area_config.network if area_config.network is vyos_defined %}
  network {{ network }} area {{ area_id }}
 {%         endfor %}
 {%         if area_config.range is vyos_defined %}
 {%             for range, range_config in area_config.range.items() %}
 {%                 if range_config.not_advertise is vyos_defined %}
  area {{ area_id }} range {{ range }} not-advertise
 {%                 else %}
  area {{ area_id }} range {{ range }}
 {%                 endif %}
 {%                 if range_config.cost is vyos_defined %}
  area {{ area_id }} range {{ range }} cost {{ range_config.cost }}
 {%                 endif %}
 {%                 if range_config.substitute is vyos_defined %}
  area {{ area_id }} range {{ range }} substitute {{ range_config.substitute }}
 {%                 endif %}
 {%             endfor %}
 {%         endif %}
 {%         if area_config.export_list is vyos_defined %}
  area {{ area_id }} export-list {{ area_config.export_list }}
 {%         endif %}
 {%         if area_config.import_list is vyos_defined %}
  area {{ area_id }} import-list {{ area_config.import_list }}
 {%         endif %}
 {%         if area_config.shortcut is vyos_defined %}
  area {{ area_id }} shortcut {{ area_config.shortcut }}
 {%         endif %}
 {%         if area_config.virtual_link is vyos_defined %}
 {%             for link, link_config in area_config.virtual_link.items() %}
 {%                 if link_config.authentication.plaintext_password is vyos_defined %}
  area {{ area_id }} virtual-link {{ link }} authentication-key {{ link_config.authentication.plaintext_password }}
 {%                 elif link_config.authentication.md5.key_id is vyos_defined %}
 {%                     for key, key_config in link_config.authentication.md5.key_id.items() %}
  area {{ area_id }} virtual-link {{ link }} message-digest-key {{ key }} md5 {{ key_config.md5_key }}
 {%                     endfor %}
 {%                 endif %}
 {#         The following values are default values #}
  area {{ area_id }} virtual-link {{ link }} hello-interval {{ link_config.hello_interval }} retransmit-interval {{ link_config.retransmit_interval }} transmit-delay {{ link_config.transmit_delay }} dead-interval {{ link_config.dead_interval }}
 {%             endfor %}
 {%         endif %}
 {%     endfor %}
 {% endif %}
 {% if auto_cost.reference_bandwidth is vyos_defined %}
  auto-cost reference-bandwidth {{ auto_cost.reference_bandwidth }}
 {% endif %}
+{% if capability.opaque is vyos_defined %}
+ capability opaque
+{% endif %}
 {% if default_information.originate is vyos_defined %}
  default-information originate {{ 'always' if default_information.originate.always is vyos_defined }} {{ 'metric ' + default_information.originate.metric if default_information.originate.metric is vyos_defined }} {{ 'metric-type ' + default_information.originate.metric_type if default_information.originate.metric_type is vyos_defined }} {{ 'route-map ' + default_information.originate.route_map if default_information.originate.route_map is vyos_defined }}
 {% endif %}
 {% if default_metric is vyos_defined %}
  default-metric {{ default_metric }}
 {% endif %}
 {% if maximum_paths is vyos_defined %}
  maximum-paths {{ maximum_paths }}
 {% endif %}
 {% if ldp_sync.holddown is vyos_defined %}
  mpls ldp-sync holddown {{ ldp_sync.holddown }}
 {% elif ldp_sync is vyos_defined %}
  mpls ldp-sync
 {% endif %}
 {% if distance.global is vyos_defined %}
  distance {{ distance.global }}
 {% endif %}
 {% if distance.ospf is vyos_defined %}
  distance ospf {{ 'intra-area ' + distance.ospf.intra_area if distance.ospf.intra_area is vyos_defined }} {{ 'inter-area ' + distance.ospf.inter_area if distance.ospf.inter_area is vyos_defined }} {{ 'external ' + distance.ospf.external if distance.ospf.external is vyos_defined }}
 {% endif %}
+{% if graceful_restart is vyos_defined %}
+{%     if graceful_restart.grace_period is vyos_defined %}
+ graceful-restart grace-period {{ graceful_restart.grace_period }}
+{%     endif %}
+{%     if graceful_restart.helper.enable.router_id is vyos_defined %}
+{%         for router_id in graceful_restart.helper.enable.router_id %}
+ graceful-restart helper enable {{ router_id }}
+{%         endfor %}
+{%     elif graceful_restart.helper.enable is vyos_defined %}
+ graceful-restart helper enable
+{%     endif %}
+{%     if graceful_restart.helper.planned_only is vyos_defined %}
+ graceful-restart helper planned-only
+{%     endif %}
+{%     if graceful_restart.helper.no_strict_lsa_checking is vyos_defined %}
+ no graceful-restart helper strict-lsa-checking
+{%     endif %}
+{%     if graceful_restart.helper.supported_grace_time is vyos_defined %}
+ graceful-restart helper supported-grace-time {{ graceful_restart.helper.supported_grace_time }}
+{%     endif %}
+{% endif %}
 {% if log_adjacency_changes is vyos_defined %}
  log-adjacency-changes {{ "detail" if log_adjacency_changes.detail is vyos_defined }}
 {% endif %}
 {% if max_metric.router_lsa.administrative is vyos_defined %}
  max-metric router-lsa administrative
 {% endif %}
 {% if max_metric.router_lsa.on_shutdown is vyos_defined %}
  max-metric router-lsa on-shutdown {{ max_metric.router_lsa.on_shutdown }}
 {% endif %}
 {% if max_metric.router_lsa.on_startup is vyos_defined %}
  max-metric router-lsa on-startup {{ max_metric.router_lsa.on_startup }}
 {% endif %}
 {% if mpls_te.enable is vyos_defined %}
  mpls-te on
  mpls-te router-address {{ mpls_te.router_address }}
 {% endif %}
 {% if neighbor is vyos_defined %}
 {%     for address, address_config in neighbor.items() %}
  neighbor {{ address }} {{ 'priority ' + address_config.priority if address_config.priority is vyos_defined }} {{ 'poll-interval ' + address_config.poll_interval if address_config.poll_interval is vyos_defined }}
 {%     endfor %}
 {% endif %}
 {% if parameters.abr_type is vyos_defined %}
  ospf abr-type {{ parameters.abr_type }}
 {% endif %}
 {% if parameters.opaque_lsa is vyos_defined %}
  ospf opaque-lsa
 {% endif %}
 {% if parameters.rfc1583_compatibility is vyos_defined %}
  ospf rfc1583compatibility
 {% endif %}
 {% if parameters.router_id is vyos_defined %}
  ospf router-id {{ parameters.router_id }}
 {% endif %}
 {% if passive_interface is vyos_defined('default') %}
  passive-interface default
 {% endif %}
 {% if redistribute is vyos_defined %}
 {%     for protocol, protocols_options in redistribute.items() %}
 {%         if protocol == 'table' %}
 {%             for table, table_options in protocols_options.items() %}
  redistribute {{ protocol }} {{ table }} {{ 'metric ' + table_options.metric if table_options.metric is vyos_defined }} {{ 'metric-type ' + table_options.metric_type if table_options.metric_type is vyos_defined }} {{ 'route-map ' + table_options.route_map if table_options.route_map is vyos_defined }}
 {%             endfor %}
 {%         else %}
  redistribute {{ protocol }} {{ 'metric ' + protocols_options.metric if protocols_options.metric is vyos_defined }} {{ 'metric-type ' + protocols_options.metric_type if protocols_options.metric_type is vyos_defined }} {{ 'route-map ' + protocols_options.route_map if protocols_options.route_map is vyos_defined }}
 {%         endif %}
 {%     endfor %}
 {% endif %}
 {% if refresh.timers is vyos_defined %}
  refresh timer {{ refresh.timers }}
 {% endif %}
 {% if summary_address is vyos_defined %}
 {%     for prefix, prefix_options in summary_address.items() %}
  summary-address {{ prefix }} {{ 'tag ' + prefix_options.tag if prefix_options.tag is vyos_defined }}{{ 'no-advertise' if prefix_options.no_advertise is vyos_defined }}
 {%     endfor %}
 {% endif %}
 {% if segment_routing is vyos_defined %}
 {%     if segment_routing.maximum_label_depth is vyos_defined %}
  segment-routing node-msd {{ segment_routing.maximum_label_depth }}
 {%     endif %}
 {%     if segment_routing.global_block is vyos_defined %}
 {%         if segment_routing.local_block is vyos_defined %}
  segment-routing global-block {{ segment_routing.global_block.low_label_value }} {{ segment_routing.global_block.high_label_value }} local-block {{ segment_routing.local_block.low_label_value }} {{ segment_routing.local_block.high_label_value }}
 {%         else %}
  segment-routing global-block {{ segment_routing.global_block.low_label_value }} {{ segment_routing.global_block.high_label_value }}
 {%         endif %}
 {%     endif %}
 {%     if segment_routing.prefix is vyos_defined %}
 {%         for prefix, prefix_config in segment_routing.prefix.items() %}
 {%             if prefix_config.index is vyos_defined %}
 {%                 if prefix_config.index.value is vyos_defined %}
  segment-routing prefix {{ prefix }} index {{ prefix_config.index.value }} {{ 'explicit-null' if prefix_config.index.explicit_null is vyos_defined }} {{ 'no-php-flag' if prefix_config.index.no_php_flag is vyos_defined }}
 {%                 endif %}
 {%             endif %}
 {%         endfor %}
 {%     endif %}
  segment-routing on
 {% endif %}
 {% if timers.throttle.spf.delay is vyos_defined and timers.throttle.spf.initial_holdtime is vyos_defined and timers.throttle.spf.max_holdtime is vyos_defined %}
 {#   Timer values have default values #}
  timers throttle spf {{ timers.throttle.spf.delay }} {{ timers.throttle.spf.initial_holdtime }} {{ timers.throttle.spf.max_holdtime }}
 {% endif %}
 exit
 !
diff --git a/interface-definitions/include/ospf/graceful-restart.xml.i b/interface-definitions/include/ospf/graceful-restart.xml.i
new file mode 100644
index 000000000..37d9a7f13
--- /dev/null
+++ b/interface-definitions/include/ospf/graceful-restart.xml.i
@@ -0,0 +1,67 @@
+<!-- include start from ospf/graceful-restart.xml.i -->
+<node name="graceful-restart">
+  <properties>
+    <help>Graceful Restart</help>
+  </properties>
+  <children>
+    <leafNode name="grace-period">
+      <properties>
+        <help>Maximum length of the grace period</help>
+        <valueHelp>
+          <format>u32:1-1800</format>
+          <description>Maximum length of the grace period in seconds</description>
+        </valueHelp>
+        <constraint>
+          <validator name="numeric" argument="--range 5-1800"/>
+        </constraint>
+      </properties>
+      <defaultValue>120</defaultValue>
+    </leafNode>
+    <node name="helper">
+      <properties>
+        <help>OSPF graceful-restart helpers</help>
+      </properties>
+      <children>
+        <node name="enable">
+          <properties>
+            <help>Enable helper support</help>
+          </properties>
+          <children>
+            <leafNode name="router-id">
+              <properties>
+                <help>Advertising Router-ID</help>
+                <valueHelp>
+                  <format>ipv4</format>
+                  <description>Router-ID in IP address format</description>
+                </valueHelp>
+                <constraint>
+                  <validator name="ipv4-address"/>
+                </constraint>
+                <multi/>
+              </properties>
+            </leafNode>
+          </children>
+        </node>
+        <leafNode name="planned-only">
+          <properties>
+            <help>Supported only planned restart</help>
+            <valueless/>
+          </properties>
+        </leafNode>
+        <leafNode name="supported-grace-time">
+          <properties>
+            <help>Supported grace timer</help>
+            <valueHelp>
+              <format>u32:10-1800</format>
+              <description>Grace interval in seconds</description>
+            </valueHelp>
+            <constraint>
+              <validator name="numeric" argument="--range 10-1800"/>
+            </constraint>
+          </properties>
+        </leafNode>
+      </children>
+    </node>
+  </children>
+</node>
+<!-- include end -->
diff --git a/interface-definitions/include/ospf/protocol-common-config.xml.i b/interface-definitions/include/ospf/protocol-common-config.xml.i
index 3492b873f..c4778e126 100644
--- a/interface-definitions/include/ospf/protocol-common-config.xml.i
+++ b/interface-definitions/include/ospf/protocol-common-config.xml.i
@@ -1,931 +1,959 @@
 <!-- include start from ospf/protocol-common-config.xml.i -->
 <node name="aggregation">
   <properties>
     <help>External route aggregation</help>
   </properties>
   <children>
     <leafNode name="timer">
       <properties>
         <help>Delay timer</help>
         <valueHelp>
           <format>u32:5-1800</format>
           <description>Timer interval in seconds</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 5-1800"/>
         </constraint>
       </properties>
       <defaultValue>5</defaultValue>
     </leafNode>
   </children>
 </node>
 <tagNode name="access-list">
   <properties>
     <help>Access list to filter networks in routing updates</help>
     <completionHelp>
       <path>policy access-list</path>
     </completionHelp>
     <valueHelp>
       <format>u32</format>
       <description>Access-list number</description>
     </valueHelp>
     <constraint>
       <validator name="numeric" argument="--range 0-4294967295"/>
     </constraint>
   </properties>
   <children>
     <leafNode name="export">
       <properties>
         <help>Filter for outgoing routing update</help>
         <completionHelp>
           <list>bgp connected kernel rip static</list>
         </completionHelp>
         <valueHelp>
           <format>bgp</format>
           <description>Filter BGP routes</description>
         </valueHelp>
         <valueHelp>
           <format>connected</format>
           <description>Filter connected routes</description>
         </valueHelp>
         <valueHelp>
           <format>isis</format>
           <description>Filter IS-IS routes</description>
         </valueHelp>
         <valueHelp>
           <format>kernel</format>
           <description>Filter Kernel routes</description>
         </valueHelp>
         <valueHelp>
           <format>rip</format>
           <description>Filter RIP routes</description>
         </valueHelp>
         <valueHelp>
           <format>static</format>
           <description>Filter static routes</description>
         </valueHelp>
         <constraint>
           <regex>(bgp|connected|isis|kernel|rip|static)</regex>
         </constraint>
         <constraintErrorMessage>Must be bgp, connected, kernel, rip, or static</constraintErrorMessage>
         <multi/>
       </properties>
     </leafNode>
   </children>
 </tagNode>
 <tagNode name="area">
   <properties>
     <help>OSPF area settings</help>
     <valueHelp>
       <format>u32</format>
       <description>OSPF area number in decimal notation</description>
     </valueHelp>
     <valueHelp>
       <format>ipv4</format>
       <description>OSPF area number in dotted decimal notation</description>
     </valueHelp>
     <constraint>
       <validator name="numeric" argument="--range 0-4294967295"/>
       <validator name="ip-address"/>
     </constraint>
   </properties>
   <children>
     <node name="area-type">
       <properties>
         <help>Area type</help>
       </properties>
       <children>
         <leafNode name="normal">
           <properties>
             <help>Normal OSPF area</help>
             <valueless/>
           </properties>
         </leafNode>
         <node name="nssa">
           <properties>
             <help>Not-So-Stubby OSPF area</help>
           </properties>
           <children>
             <leafNode name="default-cost">
               <properties>
                 <help>Summary-default cost of an NSSA area</help>
                 <valueHelp>
                   <format>u32:0-16777215</format>
                   <description>Summary default cost</description>
                 </valueHelp>
                 <constraint>
                   <validator name="numeric" argument="--range 0-16777215"/>
                 </constraint>
               </properties>
             </leafNode>
             <leafNode name="no-summary">
               <properties>
                 <help>Do not inject inter-area routes into stub</help>
                 <valueless/>
               </properties>
             </leafNode>
             <leafNode name="translate">
               <properties>
                 <help>Configure NSSA-ABR</help>
                 <completionHelp>
                   <list>always candidate never</list>
                 </completionHelp>
                 <valueHelp>
                   <format>always</format>
                   <description>Always translate LSA types</description>
                 </valueHelp>
                 <valueHelp>
                   <format>candidate</format>
                   <description>Translate for election</description>
                 </valueHelp>
                 <valueHelp>
                   <format>never</format>
                   <description>Never translate LSA types</description>
                 </valueHelp>
                 <constraint>
                   <regex>(always|candidate|never)</regex>
                 </constraint>
               </properties>
               <defaultValue>candidate</defaultValue>
             </leafNode>
           </children>
         </node>
         <node name="stub">
           <properties>
             <help>Stub OSPF area</help>
           </properties>
           <children>
             <leafNode name="default-cost">
               <properties>
                 <help>Summary-default cost</help>
                 <valueHelp>
                   <format>u32:0-16777215</format>
                   <description>Summary default cost</description>
                 </valueHelp>
                 <constraint>
                   <validator name="numeric" argument="--range 0-16777215"/>
                 </constraint>
               </properties>
             </leafNode>
             <leafNode name="no-summary">
               <properties>
                 <help>Do not inject inter-area routes into the stub</help>
                 <valueless/>
               </properties>
             </leafNode>
           </children>
         </node>
       </children>
     </node>
     <leafNode name="authentication">
       <properties>
         <help>OSPF area authentication type</help>
         <completionHelp>
           <list>plaintext-password md5</list>
         </completionHelp>
         <valueHelp>
           <format>plaintext-password</format>
           <description>Use plain-text authentication</description>
         </valueHelp>
         <valueHelp>
           <format>md5</format>
           <description>Use MD5 authentication</description>
         </valueHelp>
         <constraint>
           <regex>(plaintext-password|md5)</regex>
         </constraint>
       </properties>
     </leafNode>
     <leafNode name="network">
       <properties>
         <help>OSPF network</help>
         <valueHelp>
           <format>ipv4net</format>
           <description>OSPF network</description>
         </valueHelp>
         <constraint>
           <validator name="ipv4-prefix"/>
         </constraint>
         <multi/>
       </properties>
     </leafNode>
     <tagNode name="range">
       <properties>
         <help>Summarize routes matching a prefix (border routers only)</help>
         <valueHelp>
           <format>ipv4net</format>
           <description>Area range prefix</description>
         </valueHelp>
         <constraint>
           <validator name="ipv4-prefix"/>
         </constraint>
       </properties>
       <children>
         <leafNode name="cost">
           <properties>
             <help>Metric for this range</help>
             <valueHelp>
               <format>u32:0-16777215</format>
               <description>Metric for this range</description>
             </valueHelp>
             <constraint>
               <validator name="numeric" argument="--range 0-16777215"/>
             </constraint>
           </properties>
         </leafNode>
         <leafNode name="not-advertise">
           <properties>
             <help>Do not advertise this range</help>
             <valueless/>
           </properties>
         </leafNode>
         <leafNode name="substitute">
           <properties>
             <help>Advertise area range as another prefix</help>
             <valueHelp>
               <format>ipv4net</format>
               <description>Advertise area range as another prefix</description>
             </valueHelp>
             <constraint>
               <validator name="ipv4-prefix"/>
             </constraint>
           </properties>
         </leafNode>
       </children>
     </tagNode>
     <leafNode name="shortcut">
       <properties>
         <help>Area shortcut mode</help>
         <completionHelp>
           <list>default disable enable</list>
         </completionHelp>
         <valueHelp>
           <format>default</format>
           <description>Set default</description>
         </valueHelp>
         <valueHelp>
           <format>disable</format>
           <description>Disable shortcutting mode</description>
         </valueHelp>
         <valueHelp>
           <format>enable</format>
           <description>Enable shortcutting mode</description>
         </valueHelp>
         <constraint>
           <regex>(default|disable|enable)</regex>
         </constraint>
       </properties>
     </leafNode>
     <leafNode name="export-list">
       <properties>
         <help>Set the filter for networks announced to other areas</help>
         <completionHelp>
           <path>policy access-list</path>
         </completionHelp>
         <valueHelp>
           <format>u32</format>
           <description>Access-list number</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 0-4294967295"/>
         </constraint>
       </properties>
     </leafNode>
     <leafNode name="import-list">
       <properties>
         <help>Set the filter for networks from other areas announced</help>
         <completionHelp>
           <path>policy access-list</path>
         </completionHelp>
         <valueHelp>
           <format>u32</format>
           <description>Access-list number</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 0-4294967295"/>
         </constraint>
       </properties>
     </leafNode>
     <tagNode name="virtual-link">
       <properties>
         <help>Virtual link</help>
         <valueHelp>
           <format>ipv4</format>
           <description>OSPF area in dotted decimal notation</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 0-4294967295"/>
           <validator name="ip-address"/>
         </constraint>
       </properties>
       <children>
         #include <include/ospf/authentication.xml.i>
         #include <include/ospf/intervals.xml.i>
       </children>
     </tagNode>
   </children>
 </tagNode>
 #include <include/ospf/auto-cost.xml.i>
+<node name="capability">
+  <properties>
+    <help>Enable specific OSPF features</help>
+  </properties>
+  <children>
+    <leafNode name="opaque">
+      <properties>
+        <help>Opaque LSA</help>
+        <valueless/>
+      </properties>
+    </leafNode>
+  </children>
+</node>
 #include <include/ospf/default-information.xml.i>
 <leafNode name="default-metric">
   <properties>
     <help>Metric of redistributed routes</help>
     <valueHelp>
       <format>u32:0-16777214</format>
       <description>Metric of redistributed routes</description>
     </valueHelp>
     <constraint>
       <validator name="numeric" argument="--range 0-16777214"/>
     </constraint>
   </properties>
 </leafNode>
+#include <include/ospf/graceful-restart.xml.i>
+<node name="graceful-restart">
+  <children>
+    <node name="helper">
+      <children>
+        <leafNode name="no-strict-lsa-checking">
+          <properties>
+            <help>Disable strict LSA check</help>
+            <valueless/>
+          </properties>
+        </leafNode>
+      </children>
+    </node>
+  </children>
+</node>
 <leafNode name="maximum-paths">
   <properties>
     <help>Maximum multiple paths (ECMP)</help>
     <valueHelp>
       <format>u32:1-64</format>
       <description>Maximum multiple paths (ECMP)</description>
     </valueHelp>
     <constraint>
       <validator name="numeric" argument="--range 1-64"/>
     </constraint>
   </properties>
 </leafNode>
 #include <include/isis/ldp-sync-protocol.xml.i>
 <node name="distance">
   <properties>
     <help>Administrative distance</help>
   </properties>
   <children>
     #include <include/ospf/distance-global.xml.i>
     <node name="ospf">
       <properties>
         <help>OSPF administrative distance</help>
       </properties>
       <children>
         #include <include/ospf/distance-per-protocol.xml.i>
       </children>
     </node>
   </children>
 </node>
 <tagNode name="interface">
   <properties>
     <help>Interface configuration</help>
     <completionHelp>
       <script>${vyos_completion_dir}/list_interfaces</script>
     </completionHelp>
     <valueHelp>
       <format>txt</format>
       <description>Interface name</description>
     </valueHelp>
     <constraint>
       #include <include/constraint/interface-name.xml.i>
     </constraint>
   </properties>
   <children>
     <leafNode name="area">
       <properties>
         <help>Enable OSPF on this interface</help>
         <completionHelp>
           <path>protocols ospf area</path>
         </completionHelp>
         <valueHelp>
           <format>u32</format>
           <description>OSPF area ID as decimal notation</description>
         </valueHelp>
         <valueHelp>
           <format>ipv4</format>
           <description>OSPF area ID in IP address notation</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 0-4294967295"/>
           <validator name="ip-address"/>
         </constraint>
       </properties>
     </leafNode>
     #include <include/ospf/authentication.xml.i>
     #include <include/ospf/intervals.xml.i>
     #include <include/ospf/interface-common.xml.i>
     #include <include/isis/ldp-sync-interface.xml.i>
     <leafNode name="bandwidth">
       <properties>
         <help>Interface bandwidth (Mbit/s)</help>
         <valueHelp>
           <format>u32:1-100000</format>
           <description>Bandwidth in Megabit/sec (for calculating OSPF cost)</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 1-100000"/>
         </constraint>
       </properties>
     </leafNode>
     <leafNode name="hello-multiplier">
       <properties>
         <help>Hello multiplier factor</help>
         <valueHelp>
           <format>u32:1-10</format>
           <description>Number of Hellos to send each second</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 1-10"/>
         </constraint>
       </properties>
     </leafNode>
     <leafNode name="network">
       <properties>
         <help>Network type</help>
         <completionHelp>
           <list>broadcast non-broadcast point-to-multipoint point-to-point</list>
         </completionHelp>
         <valueHelp>
           <format>broadcast</format>
           <description>Broadcast network type</description>
         </valueHelp>
         <valueHelp>
           <format>non-broadcast</format>
           <description>Non-broadcast network type</description>
         </valueHelp>
         <valueHelp>
           <format>point-to-multipoint</format>
           <description>Point-to-multipoint network type</description>
         </valueHelp>
         <valueHelp>
           <format>point-to-point</format>
           <description>Point-to-point network type</description>
         </valueHelp>
         <constraint>
           <regex>(broadcast|non-broadcast|point-to-multipoint|point-to-point)</regex>
         </constraint>
         <constraintErrorMessage>Must be broadcast, non-broadcast, point-to-multipoint or point-to-point</constraintErrorMessage>
       </properties>
     </leafNode>
     <node name="passive">
       <properties>
         <help>Suppress routing updates on an interface</help>
       </properties>
       <children>
         #include <include/generic-disable-node.xml.i>
       </children>
     </node>
   </children>
 </tagNode>
 #include <include/ospf/log-adjacency-changes.xml.i>
 <node name="max-metric">
   <properties>
     <help>OSPF maximum and infinite-distance metric</help>
   </properties>
   <children>
     <node name="router-lsa">
       <properties>
         <help>Advertise own Router-LSA with infinite distance (stub router)</help>
       </properties>
       <children>
         <leafNode name="administrative">
           <properties>
             <help>Administratively apply, for an indefinite period</help>
             <valueless/>
           </properties>
         </leafNode>
         <leafNode name="on-shutdown">
           <properties>
             <help>Advertise stub-router prior to full shutdown of OSPF</help>
             <valueHelp>
               <format>u32:5-100</format>
               <description>Time (seconds) to advertise self as stub-router</description>
             </valueHelp>
             <constraint>
               <validator name="numeric" argument="--range 5-100"/>
             </constraint>
           </properties>
         </leafNode>
         <leafNode name="on-startup">
           <properties>
             <help>Automatically advertise stub Router-LSA on startup of OSPF</help>
             <valueHelp>
               <format>u32:5-86400</format>
               <description>Time (seconds) to advertise self as stub-router</description>
             </valueHelp>
             <constraint>
               <validator name="numeric" argument="--range 5-86400"/>
             </constraint>
           </properties>
         </leafNode>
       </children>
     </node>
   </children>
 </node>
 <node name="mpls-te">
   <properties>
     <help>MultiProtocol Label Switching-Traffic Engineering (MPLS-TE) parameters</help>
   </properties>
   <children>
     <leafNode name="enable">
       <properties>
         <help>Enable MPLS-TE functionality</help>
         <valueless/>
       </properties>
     </leafNode>
     <leafNode name="router-address">
       <properties>
         <help>Stable IP address of the advertising router</help>
         <valueHelp>
           <format>ipv4</format>
           <description>Stable IP address of the advertising router</description>
         </valueHelp>
         <constraint>
           <validator name="ipv4-address"/>
         </constraint>
       </properties>
       <defaultValue>0.0.0.0</defaultValue>
     </leafNode>
   </children>
 </node>
 <tagNode name="neighbor">
   <properties>
     <help>Specify neighbor router</help>
     <valueHelp>
       <format>ipv4</format>
       <description>Neighbor IP address</description>
     </valueHelp>
     <constraint>
       <validator name="ipv4-address"/>
     </constraint>
   </properties>
   <children>
     <leafNode name="poll-interval">
       <properties>
         <help>Dead neighbor polling interval</help>
         <valueHelp>
           <format>u32:1-65535</format>
           <description>Seconds between dead neighbor polling interval</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 1-65535"/>
         </constraint>
       </properties>
       <defaultValue>60</defaultValue>
     </leafNode>
     <leafNode name="priority">
       <properties>
         <help>Neighbor priority in seconds</help>
         <valueHelp>
           <format>u32:0-255</format>
           <description>Neighbor priority</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 0-255"/>
         </constraint>
       </properties>
       <defaultValue>0</defaultValue>
     </leafNode>
   </children>
 </tagNode>
 <node name="parameters">
   <properties>
     <help>OSPF specific parameters</help>
   </properties>
   <children>
     <leafNode name="abr-type">
       <properties>
         <help>OSPF ABR type</help>
         <completionHelp>
           <list>cisco ibm shortcut standard</list>
         </completionHelp>
         <valueHelp>
           <format>cisco</format>
           <description>Cisco ABR type</description>
         </valueHelp>
         <valueHelp>
           <format>ibm</format>
           <description>IBM ABR type</description>
         </valueHelp>
         <valueHelp>
           <format>shortcut</format>
           <description>Shortcut ABR type</description>
         </valueHelp>
         <valueHelp>
           <format>standard</format>
           <description>Standard ABR type</description>
         </valueHelp>
         <constraint>
           <regex>(cisco|ibm|shortcut|standard)</regex>
         </constraint>
       </properties>
       <defaultValue>cisco</defaultValue>
     </leafNode>
     <leafNode name="opaque-lsa">
       <properties>
         <help>Enable the Opaque-LSA capability (rfc2370)</help>
         <valueless/>
       </properties>
     </leafNode>
     <leafNode name="rfc1583-compatibility">
       <properties>
         <help>Enable RFC1583 criteria for handling AS external routes</help>
         <valueless/>
       </properties>
     </leafNode>
     #include <include/router-id.xml.i>
   </children>
 </node>
 <leafNode name="passive-interface">
   <properties>
     <help>Suppress routing updates on an interface</help>
     <completionHelp>
       <list>default</list>
     </completionHelp>
     <valueHelp>
       <format>default</format>
       <description>Default to suppress routing updates on all interfaces</description>
     </valueHelp>
     <constraint>
       <regex>(default)</regex>
     </constraint>
   </properties>
 </leafNode>
 <node name="segment-routing">
   <properties>
     <help>Segment-Routing (SPRING) settings</help>
   </properties>
   <children>
     <node name="global-block">
       <properties>
         <help>Segment Routing Global Block label range</help>
       </properties>
       <children>
         #include <include/segment-routing-label-value.xml.i>
       </children>
     </node>
     <node name="local-block">
       <properties>
         <help>Segment Routing Local Block label range</help>
       </properties>
       <children>
         #include <include/segment-routing-label-value.xml.i>
       </children>
     </node>
     <leafNode name="maximum-label-depth">
       <properties>
         <help>Maximum MPLS labels allowed for this router</help>
         <valueHelp>
           <format>u32:1-16</format>
             <description>MPLS label depth</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 1-16"/>
         </constraint>
       </properties>
     </leafNode>
     <tagNode name="prefix">
       <properties>
         <help>Static IPv4 prefix segment/label mapping</help>
         <valueHelp>
           <format>ipv4net</format>
           <description>IPv4 prefix segment</description>
         </valueHelp>
         <constraint>
           <validator name="ipv4-prefix"/>
         </constraint>
       </properties>
       <children>
         <node name="index">
           <properties>
             <help>Specify the index value of prefix segment/label ID</help>
           </properties>
           <children>
             <leafNode name="value">
               <properties>
                 <help>Specify the index value of prefix segment/label ID</help>
                 <valueHelp>
                   <format>u32:0-65535</format>
                     <description>The index segment/label ID value</description>
                 </valueHelp>
                 <constraint>
                   <validator name="numeric" argument="--range 0-65535"/>
                 </constraint>
               </properties>
             </leafNode>
             <leafNode name="explicit-null">
               <properties>
                 <help>Request upstream neighbor to replace segment/label with explicit null label</help>
                 <valueless/>
               </properties>
             </leafNode>
             <leafNode name="no-php-flag">
               <properties>
                 <help>Do not request penultimate hop popping for segment/label</help>
                 <valueless/>
               </properties>
             </leafNode>
           </children>
         </node>
       </children>
     </tagNode>
   </children>
 </node>
 <node name="redistribute">
   <properties>
     <help>Redistribute information from another routing protocol</help>
   </properties>
   <children>
     <node name="bgp">
       <properties>
         <help>Redistribute BGP routes</help>
       </properties>
       <children>
         #include <include/ospf/metric.xml.i>
         #include <include/ospf/metric-type.xml.i>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="connected">
       <properties>
         <help>Redistribute connected routes</help>
       </properties>
       <children>
         #include <include/ospf/metric.xml.i>
         #include <include/ospf/metric-type.xml.i>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="isis">
       <properties>
         <help>Redistribute IS-IS routes</help>
       </properties>
       <children>
         #include <include/ospf/metric.xml.i>
         #include <include/ospf/metric-type.xml.i>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="kernel">
       <properties>
         <help>Redistribute Kernel routes</help>
       </properties>
       <children>
         #include <include/ospf/metric.xml.i>
         #include <include/ospf/metric-type.xml.i>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="rip">
       <properties>
         <help>Redistribute RIP routes</help>
       </properties>
       <children>
         #include <include/ospf/metric.xml.i>
         #include <include/ospf/metric-type.xml.i>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="babel">
       <properties>
         <help>Redistribute Babel routes</help>
       </properties>
       <children>
         #include <include/ospf/metric.xml.i>
         #include <include/ospf/metric-type.xml.i>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="static">
       <properties>
         <help>Redistribute statically configured routes</help>
       </properties>
       <children>
         #include <include/ospf/metric.xml.i>
         #include <include/ospf/metric-type.xml.i>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <tagNode name="table">
       <properties>
         <help>Redistribute non-main Kernel Routing Table</help>
         <completionHelp>
           <path>protocols static table</path>
         </completionHelp>
         <valueHelp>
           <format>u32:1-200</format>
           <description>Policy route table number</description>
         </valueHelp>
       </properties>
       <children>
         #include <include/ospf/metric.xml.i>
         #include <include/ospf/metric-type.xml.i>
         #include <include/route-map.xml.i>
       </children>
     </tagNode>
   </children>
 </node>
 <node name="refresh">
   <properties>
     <help>Adjust refresh parameters</help>
   </properties>
   <children>
     <leafNode name="timers">
       <properties>
         <help>Refresh timer</help>
         <valueHelp>
           <format>u32:10-1800</format>
           <description>Timer value in seconds</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 10-1800"/>
         </constraint>
       </properties>
     </leafNode>
   </children>
 </node>
 <tagNode name="summary-address">
   <properties>
     <help>External summary address</help>
     <valueHelp>
       <format>ipv4net</format>
       <description>OSPF area number in dotted decimal notation</description>
     </valueHelp>
     <constraint>
       <validator name="ipv4-prefix"/>
     </constraint>
   </properties>
   <children>
     <leafNode name="no-advertise">
       <properties>
         <help>Don not advertise summary route</help>
         <valueless/>
       </properties>
     </leafNode>
     <leafNode name="tag">
       <properties>
         <help>Router tag</help>
         <valueHelp>
           <format>u32:1-4294967295</format>
           <description>Router tag value</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 1-4294967295"/>
         </constraint>
       </properties>
     </leafNode>
   </children>
 </tagNode>
 <node name="timers">
   <properties>
     <help>Adjust routing timers</help>
   </properties>
   <children>
     <node name="throttle">
       <properties>
         <help>Throttling adaptive timers</help>
       </properties>
       <children>
         <node name="spf">
           <properties>
             <help>OSPF SPF timers</help>
           </properties>
           <children>
             <leafNode name="delay">
               <properties>
                 <help>Delay from the first change received to SPF calculation</help>
                 <valueHelp>
                   <format>u32:0-600000</format>
                   <description>Delay in milliseconds</description>
                 </valueHelp>
                 <constraint>
                   <validator name="numeric" argument="--range 0-600000"/>
                 </constraint>
               </properties>
               <defaultValue>200</defaultValue>
             </leafNode>
             <leafNode name="initial-holdtime">
               <properties>
                 <help>Initial hold time between consecutive SPF calculations</help>
                 <valueHelp>
                   <format>u32:0-600000</format>
                   <description>Initial hold time in milliseconds</description>
                 </valueHelp>
                 <constraint>
                   <validator name="numeric" argument="--range 0-600000"/>
                 </constraint>
               </properties>
               <defaultValue>1000</defaultValue>
             </leafNode>
             <leafNode name="max-holdtime">
               <properties>
                 <help>Maximum hold time</help>
                 <valueHelp>
                   <format>u32:0-600000</format>
                   <description>Max hold time in milliseconds</description>
                 </valueHelp>
                 <constraint>
                   <validator name="numeric" argument="--range 0-600000"/>
                 </constraint>
               </properties>
               <defaultValue>10000</defaultValue>
             </leafNode>
           </children>
         </node>
       </children>
     </node>
   </children>
 </node>
-<!-- include end -->
\ No newline at end of file
+<!-- include end -->
diff --git a/interface-definitions/include/ospfv3/protocol-common-config.xml.i b/interface-definitions/include/ospfv3/protocol-common-config.xml.i
index a7de50638..4c3ca68e1 100644
--- a/interface-definitions/include/ospfv3/protocol-common-config.xml.i
+++ b/interface-definitions/include/ospfv3/protocol-common-config.xml.i
@@ -1,259 +1,274 @@
 <!-- include start from ospfv3/protocol-common-config.xml.i -->
 <tagNode name="area">
   <properties>
     <help>OSPFv3 Area</help>
     <valueHelp>
       <format>u32</format>
       <description>Area ID as a decimal value</description>
     </valueHelp>
     <valueHelp>
       <format>ipv4</format>
       <description>Area ID in IP address forma</description>
     </valueHelp>
     <constraint>
       <validator name="numeric" argument="--range 0-4294967295"/>
       <validator name="ip-address"/>
     </constraint>
   </properties>
   <children>
     <node name="area-type">
       <properties>
         <help>OSPFv3 Area type</help>
       </properties>
       <children>
         <node name="nssa">
           <properties>
             <help>NSSA OSPFv3 area</help>
           </properties>
           <children>
             <leafNode name="default-information-originate">
               <properties>
                 <help>Originate Type 7 default into NSSA area</help>
                 <valueless/>
               </properties>
             </leafNode>
             #include <include/ospfv3/no-summary.xml.i>
           </children>
         </node>
         <node name="stub">
           <properties>
             <help>Stub OSPFv3 area</help>
           </properties>
           <children>
             #include <include/ospfv3/no-summary.xml.i>
           </children>
         </node>
       </children>
     </node>
     <leafNode name="export-list">
       <properties>
         <help>Name of export-list</help>
         <completionHelp>
           <path>policy access-list6</path>
         </completionHelp>
       </properties>
     </leafNode>
     <leafNode name="import-list">
       <properties>
         <help>Name of import-list</help>
         <completionHelp>
           <path>policy access-list6</path>
         </completionHelp>
       </properties>
     </leafNode>
     <tagNode name="range">
       <properties>
         <help>Specify IPv6 prefix (border routers only)</help>
         <valueHelp>
           <format>ipv6net</format>
           <description>Specify IPv6 prefix (border routers only)</description>
         </valueHelp>
         <constraint>
           <validator name="ipv6-prefix"/>
         </constraint>
       </properties>
       <children>
         <leafNode name="advertise">
           <properties>
             <help>Advertise this range</help>
             <valueless/>
           </properties>
         </leafNode>
         <leafNode name="not-advertise">
           <properties>
             <help>Do not advertise this range</help>
             <valueless/>
           </properties>
         </leafNode>
       </children>
     </tagNode>
   </children>
 </tagNode>
 #include <include/ospf/auto-cost.xml.i>
 #include <include/ospf/default-information.xml.i>
 <node name="distance">
   <properties>
     <help>Administrative distance</help>
   </properties>
   <children>
     #include <include/ospf/distance-global.xml.i>
     <node name="ospfv3">
       <properties>
         <help>OSPFv3 administrative distance</help>
       </properties>
       <children>
         #include <include/ospf/distance-per-protocol.xml.i>
       </children>
     </node>
   </children>
 </node>
+#include <include/ospf/graceful-restart.xml.i>
+<node name="graceful-restart">
+  <children>
+    <node name="helper">
+      <children>
+        <leafNode name="lsa-check-disable">
+          <properties>
+            <help>Disable strict LSA check</help>
+            <valueless/>
+          </properties>
+        </leafNode>
+      </children>
+    </node>
+  </children>
+</node>
 <tagNode name="interface">
   <properties>
     <help>Enable routing on an IPv6 interface</help>
     <completionHelp>
       <script>${vyos_completion_dir}/list_interfaces</script>
     </completionHelp>
     <valueHelp>
       <format>txt</format>
       <description>Interface used for routing information exchange</description>
     </valueHelp>
     <constraint>
       #include <include/constraint/interface-name.xml.i>
     </constraint>
   </properties>
   <children>
     <leafNode name="area">
       <properties>
         <help>Enable OSPF on this interface</help>
         <completionHelp>
           <path>protocols ospfv3 area</path>
         </completionHelp>
         <valueHelp>
           <format>u32</format>
           <description>OSPF area ID as decimal notation</description>
         </valueHelp>
         <valueHelp>
           <format>ipv4</format>
           <description>OSPF area ID in IP address notation</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 0-4294967295"/>
           <validator name="ip-address"/>
         </constraint>
       </properties>
     </leafNode>
     #include <include/ospf/intervals.xml.i>
     #include <include/ospf/interface-common.xml.i>
     <leafNode name="ifmtu">
       <properties>
         <help>Interface MTU</help>
         <valueHelp>
           <format>u32:1-65535</format>
           <description>Interface MTU</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 1-65535"/>
         </constraint>
       </properties>
     </leafNode>
     <leafNode name="instance-id">
       <properties>
         <help>Instance ID</help>
         <valueHelp>
           <format>u32:0-255</format>
           <description>Instance Id</description>
         </valueHelp>
         <constraint>
           <validator name="numeric" argument="--range 0-255"/>
         </constraint>
       </properties>
       <defaultValue>0</defaultValue>
     </leafNode>
     <leafNode name="network">
       <properties>
         <help>Network type</help>
         <completionHelp>
           <list>broadcast point-to-point</list>
         </completionHelp>
         <valueHelp>
           <format>broadcast</format>
           <description>Broadcast network type</description>
         </valueHelp>
         <valueHelp>
           <format>point-to-point</format>
           <description>Point-to-point network type</description>
         </valueHelp>
         <constraint>
           <regex>(broadcast|point-to-point)</regex>
         </constraint>
         <constraintErrorMessage>Must be broadcast or point-to-point</constraintErrorMessage>
       </properties>
     </leafNode>
     #include <include/isis/passive.xml.i>
   </children>
 </tagNode>
 #include <include/ospf/log-adjacency-changes.xml.i>
 <node name="parameters">
   <properties>
     <help>OSPFv3 specific parameters</help>
   </properties>
   <children>
     #include <include/router-id.xml.i>
   </children>
 </node>
 <node name="redistribute">
   <properties>
     <help>Redistribute information from another routing protocol</help>
   </properties>
   <children>
     <node name="bgp">
       <properties>
         <help>Redistribute BGP routes</help>
       </properties>
       <children>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="connected">
       <properties>
         <help>Redistribute connected routes</help>
       </properties>
       <children>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="kernel">
       <properties>
         <help>Redistribute kernel routes</help>
       </properties>
       <children>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="ripng">
       <properties>
         <help>Redistribute RIPNG routes</help>
       </properties>
       <children>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="babel">
       <properties>
         <help>Redistribute Babel routes</help>
       </properties>
       <children>
         #include <include/route-map.xml.i>
       </children>
     </node>
     <node name="static">
       <properties>
         <help>Redistribute static routes</help>
       </properties>
       <children>
         #include <include/route-map.xml.i>
       </children>
     </node>
   </children>
 </node>
 <!-- include end -->
diff --git a/op-mode-definitions/include/ospf-common.xml.i b/op-mode-definitions/include/ospf/common.xml.i
similarity index 99%
rename from op-mode-definitions/include/ospf-common.xml.i
rename to op-mode-definitions/include/ospf/common.xml.i
index 979ffb07e..c8341bd3e 100644
--- a/op-mode-definitions/include/ospf-common.xml.i
+++ b/op-mode-definitions/include/ospf/common.xml.i
@@ -1,559 +1,559 @@
 <!-- included start from ospf-common.xml.i -->
 <leafNode name="border-routers">
   <properties>
     <help>Show IPv4 OSPF border-routers information</help>
   </properties>
   <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
 </leafNode>
 <node name="database">
   <properties>
     <help>Show IPv4 OSPF database information</help>
   </properties>
   <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
   <children>
     <node name="asbr-summary">
       <properties>
         <help>Show IPv4 OSPF ASBR summary database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF ASBR summary database for given address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF ASBR summary database for given address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="asbr-summary">
       <properties>
         <help>Show IPv4 OSPF ASBR summary database information of given address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF ASBR summary database of given address for given advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show summary of self-originate IPv4 OSPF ASBR database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
     <node name="external">
       <properties>
         <help>Show IPv4 OSPF external database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF external database for specified IP address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF external database for specified IP address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="external">
       <properties>
         <help>Show IPv4 OSPF external database information of specified IP address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF external database of specified IP address for specified advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show self-originate IPv4 OSPF external database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
     <leafNode name="max-age">
       <properties>
         <help>Show IPv4 OSPF max-age database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
     </leafNode>
     <node name="network">
       <properties>
         <help>Show IPv4 OSPF network database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF network database for specified IP address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF network database for given address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="network">
       <properties>
         <help>Show IPv4 OSPF network database information of specified IP address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF network database of specified IP address for specified advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show self-originate IPv4 OSPF network database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
     <node name="nssa-external">
       <properties>
         <help>Show IPv4 OSPF NSSA external database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF NSSA external database for specified IP address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF NSSA external database for specified IP address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="nssa-external">
       <properties>
         <help>Show IPv4 OSPF NSSA external database information of specified IP address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF NSSA external database of specified IP address for specified advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show self-originate IPv4 OSPF NSSA external database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
     <node name="opaque-area">
       <properties>
         <help>Show IPv4 OSPF opaque-area database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-area database for specified IP address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-area database for specified IP address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="opaque-area">
       <properties>
         <help>Show IPv4 OSPF opaque-area database information of specified IP address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-area database of specified IP address for specified advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show self-originate IPv4 OSPF opaque-area database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
     <node name="opaque-as">
       <properties>
         <help>Show IPv4 OSPF opaque-as database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-as database for specified IP address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-as database for specified IP address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="opaque-as">
       <properties>
         <help>Show IPv4 OSPF opaque-as database information of specified IP address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-as database of specified IP address for specified advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show self-originate IPv4 OSPF opaque-as database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
     <node name="opaque-link">
       <properties>
         <help>Show IPv4 OSPF opaque-link database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-link database for specified IP address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-link database for specified IP address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="opaque-link">
       <properties>
         <help>Show IPv4 OSPF opaque-link database information of specified IP address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF opaque-link database of specified IP address for specified advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show self-originate IPv4 OSPF opaque-link database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
     <node name="router">
       <properties>
         <help>Show IPv4 OSPF router database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF router database for specified IP address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF router database for specified IP address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="router">
       <properties>
         <help>Show IPv4 OSPF router database information of specified IP address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF router database of specified IP address for specified advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show self-originate IPv4 OSPF router database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
     <leafNode name="self-originate">
       <properties>
         <help>Show IPv4 OSPF self-originate database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
     </leafNode>
     <node name="summary">
       <properties>
         <help>Show summary of IPv4 OSPF database</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF summary database for specified IP address of advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <node name="adv-router">
           <properties>
             <help>Show IPv4 OSPF summary database for specified IP address of advertised router</help>
           </properties>
         </node>
       </children>
     </node>
     <tagNode name="summary">
       <properties>
         <help>Show IPv4 OSPF summary database information of specified IP address</help>
         <completionHelp>
           <list>&lt;x.x.x.x&gt;</list>
         </completionHelp>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
       <children>
         <node name="adv-router">
           <properties>
             <help>Show advertising router link states</help>
           </properties>
         </node>
         <tagNode name="adv-router">
           <properties>
             <help>Show IPv4 OSPF summary database of specified IP address for specified advertised router</help>
             <completionHelp>
              <list>&lt;x.x.x.x&gt;</list>
             </completionHelp>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </tagNode>
         <leafNode name="self-originate">
           <properties>
             <help>Show self-originate IPv4 OSPF summary database</help>
           </properties>
           <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
         </leafNode>
       </children>
     </tagNode>
   </children>
 </node>
+#include <include/ospf/graceful-restart.xml.i>
 <node name="interface">
   <properties>
     <help>Show IPv4 OSPF interface information</help>
   </properties>
   <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
 </node>
 #include <include/vtysh-generic-interface-tagNode.xml.i>
 <node name="mpls">
   <properties>
     <help>Show MPLS information</help>
   </properties>
   <children>
   #include <include/ldp-sync.xml.i>
   </children>
 </node>
 <node name="neighbor">
   <properties>
     <help>Show IPv4 OSPF neighbor information</help>
   </properties>
   <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
   <children>
     <node name="detail">
       <properties>
         <help>Show detailed IPv4 OSPF neighbor information</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
     </node>
   </children>
 </node>
 <tagNode name="neighbor">
   <properties>
     <help>Show IPv4 OSPF neighbor information for specified IP address or interface</help>
     <completionHelp>
       <list>&lt;x.x.x.x&gt;</list>
       <script>${vyos_completion_dir}/list_interfaces</script>
     </completionHelp>
   </properties>
   <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
 </tagNode>
 <node name="route">
   <properties>
     <help>Show IPv4 OSPF route information</help>
   </properties>
   <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
   <children>
     <leafNode name="detail">
       <properties>
         <help>Show detailed IPv4 OSPF route information</help>
       </properties>
       <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
     </leafNode>
   </children>
 </node>
 <!-- included end -->
-
diff --git a/op-mode-definitions/include/ospf/graceful-restart.xml.i b/op-mode-definitions/include/ospf/graceful-restart.xml.i
new file mode 100644
index 000000000..736d8f951
--- /dev/null
+++ b/op-mode-definitions/include/ospf/graceful-restart.xml.i
@@ -0,0 +1,13 @@
+<node name="graceful-restart">
+    <properties>
+      <help>Show IPv4 OSPF Graceful Restart</help>
+    </properties>
+    <children>
+      <leafNode name="helper">
+        <properties>
+          <help>OSPF Graceful Restart helper details</help>
+        </properties>
+        <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
+      </leafNode>
+    </children>
+  </node>
diff --git a/op-mode-definitions/show-ip-ospf.xml.in b/op-mode-definitions/show-ip-ospf.xml.in
index 704ed984f..f3b9da90c 100644
--- a/op-mode-definitions/show-ip-ospf.xml.in
+++ b/op-mode-definitions/show-ip-ospf.xml.in
@@ -1,36 +1,36 @@
 <?xml version="1.0"?>
 <interfaceDefinition>
   <node name="show">
     <children>
       <node name="ip">
         <properties>
           <help>Show IPv4 routing information</help>
         </properties>
         <children>
           <node name="ospf">
             <properties>
               <help>Show IPv4 Open Shortest Path First (OSPF) routing information</help>
             </properties>
             <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
             <children>
-              #include <include/ospf-common.xml.i>
+              #include <include/ospf/common.xml.i>
               <tagNode name="vrf">
                 <properties>
                   <help>Show OSPF routing protocol for given VRF</help>
                   <completionHelp>
                     <path>vrf name</path>
                     <list>all</list>
                   </completionHelp>
                 </properties>
                 <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
                 <children>
-                  #include <include/ospf-common.xml.i>
+                  #include <include/ospf/common.xml.i>
                 </children>
               </tagNode>
             </children>
           </node>
         </children>
       </node>
     </children>
   </node>
 </interfaceDefinition>
diff --git a/op-mode-definitions/show-ipv6-ospfv3.xml.in b/op-mode-definitions/show-ipv6-ospfv3.xml.in
index a63465472..e1fcf470f 100644
--- a/op-mode-definitions/show-ipv6-ospfv3.xml.in
+++ b/op-mode-definitions/show-ipv6-ospfv3.xml.in
@@ -1,116 +1,118 @@
 <?xml version="1.0"?>
 <interfaceDefinition>
   <node name="show">
     <children>
       <node name="ipv6">
         <properties>
           <help>Show IPv6 routing information</help>
         </properties>
         <children>
           <node name="ospfv3">
             <properties>
               <help>Show IPv6 Open Shortest Path First (OSPF)</help>
             </properties>
             <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
             <children>
               <node name="area">
                 <properties>
                   <help>Show Shortest Path First tree information</help>
                 </properties>
                 <command>vtysh -c "show ipv6 ospf6 spf tree"</command>
               </node>
               <tagNode name="area">
                 <properties>
                   <help>Area ID (as an IPv4 notation)</help>
                   <completionHelp>
                     <path>protocols ospfv3 area</path>
                   </completionHelp>
                 </properties>
                 <command>vtysh -c "show ipv6 ospf6 area $5 spf tree"</command>
                 <children>
                   <tagNode name="router">
                     <properties>
                       <help> Simulate view point (Router ID)</help>
                       <completionHelp>
                         <list>&lt;x.x.x.x&gt;</list>
                       </completionHelp>
                     </properties>
                     <command>vtysh -c "show ipv6 ospf6 simulate spf-tree $7 $4 $5"</command>
                   </tagNode>
                 </children>
               </tagNode>
               #include <include/ospfv3/border-routers.xml.i>
               #include <include/ospfv3/database.xml.i>
+              #include <include/ospf/graceful-restart.xml.i>
               #include <include/ospfv3/interface.xml.i>
               #include <include/ospfv3/linkstate.xml.i>
               #include <include/ospfv3/neighbor.xml.i>
               #include <include/ospfv3/redistribute.xml.i>
               #include <include/ospfv3/route.xml.i>
               <node name="vrf">
                 <properties>
                   <help>Specify the VRF</help>
                   <completionHelp>
                     <list>all</list>
                     <path>vrf name</path>
                   </completionHelp>
                 </properties>
                 <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
               </node>
               <tagNode name="vrf">
                 <properties>
                   <help>VRF name</help>
                   <completionHelp>
                     <list>all</list>
                     <path>vrf name</path>
                   </completionHelp>
                 </properties>
                 <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
                 <children>
                   <node name="area">
                     <properties>
                       <help>Show Shortest Path First tree information</help>
                     </properties>
                     <command>vtysh -c "show ipv6 ospf6 vrf $5 spf tree"</command>
                   </node>
                   <tagNode name="area">
                     <properties>
                       <help>Area ID (as an IPv4 notation)</help>
                       <completionHelp>
                         <path>protocols ospfv3 area</path>
                       </completionHelp>
                     </properties>
                     <command>vtysh -c "show ipv6 ospf6 vrf $5 area $7 spf tree"</command>
                     <children>
                       <tagNode name="router">
                         <properties>
                           <help> Simulate view point (Router ID)</help>
                           <completionHelp>
                             <list>&lt;x.x.x.x&gt;</list>
                           </completionHelp>
                         </properties>
                         <command>vtysh -c "show ipv6 ospf6 vrf $5 simulate spf-tree $9 $6 $7"</command>
                       </tagNode>
                     </children>
                   </tagNode>
                   #include <include/ospfv3/border-routers.xml.i>
                   #include <include/ospfv3/database.xml.i>
+                  #include <include/ospf/graceful-restart.xml.i>
                   #include <include/ospfv3/interface.xml.i>
                   #include <include/ospfv3/linkstate.xml.i>
                   #include <include/ospfv3/neighbor.xml.i>
                   #include <include/ospfv3/redistribute.xml.i>
                   #include <include/ospfv3/route.xml.i>
                 </children>
               </tagNode>
               <leafNode name="vrfs">
                 <properties>
                   <help>Show OSPFv3 VRFs</help>
                 </properties>
                 <command>${vyos_op_scripts_dir}/vtysh_wrapper.sh $@</command>
               </leafNode>
             </children>
           </node>
         </children>
       </node>
     </children>
   </node>
 </interfaceDefinition>
diff --git a/smoketest/scripts/cli/test_protocols_ospf.py b/smoketest/scripts/cli/test_protocols_ospf.py
index 977376bdd..80befbfd6 100755
--- a/smoketest/scripts/cli/test_protocols_ospf.py
+++ b/smoketest/scripts/cli/test_protocols_ospf.py
@@ -1,483 +1,510 @@
 #!/usr/bin/env python3
 #
 # Copyright (C) 2021-2022 VyOS maintainers and contributors
 #
 # This program is free software; you can redistribute it and/or modify
 # it under the terms of the GNU General Public License version 2 or later as
 # published by the Free Software Foundation.
 #
 # This program is distributed in the hope that it will be useful,
 # but WITHOUT ANY WARRANTY; without even the implied warranty of
 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 # GNU General Public License for more details.
 #
 # You should have received a copy of the GNU General Public License
 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
 
 import unittest
 
 from base_vyostest_shim import VyOSUnitTestSHIM
 
 from vyos.configsession import ConfigSessionError
 from vyos.ifconfig import Section
 from vyos.utils.process import process_named_running
 
 PROCESS_NAME = 'ospfd'
 base_path = ['protocols', 'ospf']
 
 route_map = 'foo-bar-baz10'
 
 class TestProtocolsOSPF(VyOSUnitTestSHIM.TestCase):
     @classmethod
     def setUpClass(cls):
         super(TestProtocolsOSPF, cls).setUpClass()
 
         cls.cli_set(cls, ['policy', 'route-map', route_map, 'rule', '10', 'action', 'permit'])
         cls.cli_set(cls, ['policy', 'route-map', route_map, 'rule', '20', 'action', 'permit'])
 
         # ensure we can also run this test on a live system - so lets clean
         # out the current configuration :)
         cls.cli_delete(cls, base_path)
 
     @classmethod
     def tearDownClass(cls):
         cls.cli_delete(cls, ['policy', 'route-map', route_map])
         super(TestProtocolsOSPF, cls).tearDownClass()
 
     def tearDown(self):
         # Check for running process
         self.assertTrue(process_named_running(PROCESS_NAME))
         self.cli_delete(base_path)
         self.cli_commit()
 
     def test_ospf_01_defaults(self):
         # commit changes
         self.cli_set(base_path)
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' auto-cost reference-bandwidth 100', frrconfig)
         self.assertIn(f' timers throttle spf 200 1000 10000', frrconfig) # defaults
 
     def test_ospf_02_simple(self):
         router_id = '127.0.0.1'
         abr_type = 'ibm'
         bandwidth = '1000'
         metric = '123'
 
         self.cli_set(base_path + ['auto-cost', 'reference-bandwidth', bandwidth])
         self.cli_set(base_path + ['parameters', 'router-id', router_id])
         self.cli_set(base_path + ['parameters', 'abr-type', abr_type])
         self.cli_set(base_path + ['parameters', 'opaque-lsa'])
         self.cli_set(base_path + ['parameters', 'rfc1583-compatibility'])
         self.cli_set(base_path + ['log-adjacency-changes', 'detail'])
         self.cli_set(base_path + ['default-metric', metric])
         self.cli_set(base_path + ['passive-interface', 'default'])
         self.cli_set(base_path + ['area', '10', 'area-type', 'stub'])
         self.cli_set(base_path + ['area', '10', 'network', '10.0.0.0/16'])
         self.cli_set(base_path + ['area', '10', 'range', '10.0.1.0/24'])
         self.cli_set(base_path + ['area', '10', 'range', '10.0.2.0/24', 'not-advertise'])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' compatible rfc1583', frrconfig)
         self.assertIn(f' auto-cost reference-bandwidth {bandwidth}', frrconfig)
         self.assertIn(f' ospf router-id {router_id}', frrconfig)
         self.assertIn(f' ospf abr-type {abr_type}', frrconfig)
         self.assertIn(f' timers throttle spf 200 1000 10000', frrconfig) # defaults
         self.assertIn(f' capability opaque', frrconfig)
         self.assertIn(f' default-metric {metric}', frrconfig)
         self.assertIn(f' passive-interface default', frrconfig)
         self.assertIn(f' area 10 stub', frrconfig)
         self.assertIn(f' network 10.0.0.0/16 area 10', frrconfig)
         self.assertIn(f' area 10 range 10.0.1.0/24', frrconfig)
         self.assertNotIn(f' area 10 range 10.0.1.0/24 not-advertise', frrconfig)
         self.assertIn(f' area 10 range 10.0.2.0/24 not-advertise', frrconfig)
 
 
     def test_ospf_03_access_list(self):
         acl = '100'
         seq = '10'
         protocols = ['bgp', 'connected', 'isis', 'kernel', 'rip', 'static']
 
         self.cli_set(['policy', 'access-list', acl, 'rule', seq, 'action', 'permit'])
         self.cli_set(['policy', 'access-list', acl, 'rule', seq, 'source', 'any'])
         self.cli_set(['policy', 'access-list', acl, 'rule', seq, 'destination', 'any'])
         for ptotocol in protocols:
             self.cli_set(base_path + ['access-list', acl, 'export', ptotocol])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' timers throttle spf 200 1000 10000', frrconfig) # defaults
         for ptotocol in protocols:
             self.assertIn(f' distribute-list {acl} out {ptotocol}', frrconfig) # defaults
         self.cli_delete(['policy', 'access-list', acl])
 
 
     def test_ospf_04_default_originate(self):
         seq = '100'
         metric = '50'
         metric_type = '1'
 
         self.cli_set(base_path + ['default-information', 'originate', 'metric', metric])
         self.cli_set(base_path + ['default-information', 'originate', 'metric-type', metric_type])
         self.cli_set(base_path + ['default-information', 'originate', 'route-map', route_map])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' timers throttle spf 200 1000 10000', frrconfig) # defaults
         self.assertIn(f' default-information originate metric {metric} metric-type {metric_type} route-map {route_map}', frrconfig)
 
         # Now set 'always'
         self.cli_set(base_path + ['default-information', 'originate', 'always'])
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f' default-information originate always metric {metric} metric-type {metric_type} route-map {route_map}', frrconfig)
 
 
     def test_ospf_05_options(self):
         global_distance = '128'
         intra_area = '100'
         inter_area = '110'
         external = '120'
         on_startup = '30'
         on_shutdown = '60'
         refresh = '50'
         aggregation_timer = '100'
         summary_nets = {
             '10.0.1.0/24' : {},
             '10.0.2.0/24' : {'tag' : '50'},
             '10.0.3.0/24' : {'no_advertise' : {}},
          }
 
         self.cli_set(base_path + ['distance', 'global', global_distance])
         self.cli_set(base_path + ['distance', 'ospf', 'external', external])
         self.cli_set(base_path + ['distance', 'ospf', 'intra-area', intra_area])
 
         self.cli_set(base_path + ['max-metric', 'router-lsa', 'on-startup', on_startup])
         self.cli_set(base_path + ['max-metric', 'router-lsa', 'on-shutdown', on_shutdown])
 
         self.cli_set(base_path + ['mpls-te', 'enable'])
         self.cli_set(base_path + ['refresh', 'timers', refresh])
 
         self.cli_set(base_path + ['aggregation', 'timer', aggregation_timer])
 
         for summary, summary_options in summary_nets.items():
             self.cli_set(base_path + ['summary-address', summary])
             if 'tag' in summary_options:
                 self.cli_set(base_path + ['summary-address', summary, 'tag', summary_options['tag']])
             if 'no_advertise' in summary_options:
                 self.cli_set(base_path + ['summary-address', summary, 'no-advertise'])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' mpls-te on', frrconfig)
         self.assertIn(f' mpls-te router-address 0.0.0.0', frrconfig) # default
         self.assertIn(f' distance {global_distance}', frrconfig)
         self.assertIn(f' distance ospf intra-area {intra_area} external {external}', frrconfig)
         self.assertIn(f' max-metric router-lsa on-startup {on_startup}', frrconfig)
         self.assertIn(f' max-metric router-lsa on-shutdown {on_shutdown}', frrconfig)
         self.assertIn(f' refresh timer {refresh}', frrconfig)
 
         self.assertIn(f' aggregation timer {aggregation_timer}', frrconfig)
         for summary, summary_options in summary_nets.items():
             self.assertIn(f' summary-address {summary}', frrconfig)
             if 'tag' in summary_options:
                 tag = summary_options['tag']
                 self.assertIn(f' summary-address {summary} tag {tag}', frrconfig)
             if 'no_advertise' in summary_options:
                 self.assertIn(f' summary-address {summary} no-advertise', frrconfig)
 
         # enable inter-area
         self.cli_set(base_path + ['distance', 'ospf', 'inter-area', inter_area])
         self.cli_commit()
 
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f' distance ospf intra-area {intra_area} inter-area {inter_area} external {external}', frrconfig)
 
 
     def test_ospf_06_neighbor(self):
         priority = '10'
         poll_interval = '20'
         neighbors = ['1.1.1.1', '2.2.2.2', '3.3.3.3']
         for neighbor in neighbors:
             self.cli_set(base_path + ['neighbor', neighbor, 'priority', priority])
             self.cli_set(base_path + ['neighbor', neighbor, 'poll-interval', poll_interval])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         for neighbor in neighbors:
             self.assertIn(f' neighbor {neighbor} priority {priority} poll-interval {poll_interval}', frrconfig) # default
 
     def test_ospf_07_redistribute(self):
         metric = '15'
         metric_type = '1'
         redistribute = ['bgp', 'connected', 'isis', 'kernel', 'rip', 'static']
 
         for protocol in redistribute:
             self.cli_set(base_path + ['redistribute', protocol, 'metric', metric])
             self.cli_set(base_path + ['redistribute', protocol, 'route-map', route_map])
             self.cli_set(base_path + ['redistribute', protocol, 'metric-type', metric_type])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         for protocol in redistribute:
             self.assertIn(f' redistribute {protocol} metric {metric} metric-type {metric_type} route-map {route_map}', frrconfig)
 
     def test_ospf_08_virtual_link(self):
         networks = ['10.0.0.0/8', '172.16.0.0/12', '192.168.0.0/16']
         area = '10'
         shortcut = 'enable'
         virtual_link = '192.0.2.1'
         hello = '6'
         retransmit = '5'
         transmit = '5'
         dead = '40'
 
         self.cli_set(base_path + ['area', area, 'shortcut', shortcut])
         self.cli_set(base_path + ['area', area, 'virtual-link', virtual_link, 'hello-interval', hello])
         self.cli_set(base_path + ['area', area, 'virtual-link', virtual_link, 'retransmit-interval', retransmit])
         self.cli_set(base_path + ['area', area, 'virtual-link', virtual_link, 'transmit-delay', transmit])
         self.cli_set(base_path + ['area', area, 'virtual-link', virtual_link, 'dead-interval', dead])
         for network in networks:
             self.cli_set(base_path + ['area', area, 'network', network])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' area {area} shortcut {shortcut}', frrconfig)
         self.assertIn(f' area {area} virtual-link {virtual_link} hello-interval {hello} retransmit-interval {retransmit} transmit-delay {transmit} dead-interval {dead}', frrconfig)
         for network in networks:
             self.assertIn(f' network {network} area {area}', frrconfig)
 
 
     def test_ospf_09_interface_configuration(self):
         interfaces = Section.interfaces('ethernet')
         password = 'vyos1234'
         bandwidth = '10000'
         cost = '150'
         network = 'point-to-point'
         priority = '200'
         bfd_profile = 'vyos-test'
 
         self.cli_set(base_path + ['passive-interface', 'default'])
         for interface in interfaces:
             base_interface = base_path + ['interface', interface]
             self.cli_set(base_interface + ['authentication', 'plaintext-password', password])
             self.cli_set(base_interface + ['bandwidth', bandwidth])
             self.cli_set(base_interface + ['bfd', 'profile', bfd_profile])
             self.cli_set(base_interface + ['cost', cost])
             self.cli_set(base_interface + ['mtu-ignore'])
             self.cli_set(base_interface + ['network', network])
             self.cli_set(base_interface + ['priority', priority])
             self.cli_set(base_interface + ['passive', 'disable'])
 
         # commit changes
         self.cli_commit()
 
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' passive-interface default', frrconfig)
 
         for interface in interfaces:
             config = self.getFRRconfig(f'interface {interface}')
             self.assertIn(f'interface {interface}', config)
             self.assertIn(f' ip ospf authentication-key {password}', config)
             self.assertIn(f' ip ospf bfd', config)
             self.assertIn(f' ip ospf bfd profile {bfd_profile}', config)
             self.assertIn(f' ip ospf cost {cost}', config)
             self.assertIn(f' ip ospf mtu-ignore', config)
             self.assertIn(f' ip ospf network {network}', config)
             self.assertIn(f' ip ospf priority {priority}', config)
             self.assertIn(f' no ip ospf passive', config)
             self.assertIn(f' bandwidth {bandwidth}', config)
 
     def test_ospf_11_interface_area(self):
         area = '0'
         interfaces = Section.interfaces('ethernet')
 
         self.cli_set(base_path + ['area', area, 'network', '10.0.0.0/8'])
         for interface in interfaces:
             self.cli_set(base_path + ['interface', interface, 'area', area])
 
         # we can not have bot area network and interface area set
         with self.assertRaises(ConfigSessionError):
             self.cli_commit()
         self.cli_delete(base_path + ['area', area, 'network'])
 
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
 
         for interface in interfaces:
             config = self.getFRRconfig(f'interface {interface}')
             self.assertIn(f'interface {interface}', config)
             self.assertIn(f' ip ospf area {area}', config)
 
     def test_ospf_12_vrfs(self):
         # It is safe to assume that when the basic VRF test works, all
         # other OSPF related features work, as we entirely inherit the CLI
         # templates and Jinja2 FRR template.
         table = '1000'
         vrf = 'blue'
         vrf_base = ['vrf', 'name', vrf]
         vrf_iface = 'eth1'
         self.cli_set(vrf_base + ['table', table])
         self.cli_set(vrf_base + ['protocols', 'ospf', 'interface', vrf_iface])
         self.cli_set(['interfaces', 'ethernet', vrf_iface, 'vrf', vrf])
 
         # Also set a default VRF OSPF config
         self.cli_set(base_path)
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' auto-cost reference-bandwidth 100', frrconfig)
         self.assertIn(f' timers throttle spf 200 1000 10000', frrconfig) # defaults
 
         frrconfig = self.getFRRconfig(f'router ospf vrf {vrf}')
         self.assertIn(f'router ospf vrf {vrf}', frrconfig)
         self.assertIn(f' auto-cost reference-bandwidth 100', frrconfig)
         self.assertIn(f' timers throttle spf 200 1000 10000', frrconfig) # defaults
 
         # cleanup
         self.cli_delete(['vrf', 'name', vrf])
         self.cli_delete(['interfaces', 'ethernet', vrf_iface, 'vrf'])
 
     def test_ospf_13_export_list(self):
         # Verify explort-list works on ospf-area
         acl = '100'
         seq = '10'
         area = '0.0.0.10'
         network = '10.0.0.0/8'
 
 
         self.cli_set(['policy', 'access-list', acl, 'rule', seq, 'action', 'permit'])
         self.cli_set(['policy', 'access-list', acl, 'rule', seq, 'source', 'any'])
         self.cli_set(['policy', 'access-list', acl, 'rule', seq, 'destination', 'any'])
         self.cli_set(base_path + ['area', area, 'network', network])
         self.cli_set(base_path + ['area', area, 'export-list', acl])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' timers throttle spf 200 1000 10000', frrconfig) # default
         self.assertIn(f' network {network} area {area}', frrconfig)
         self.assertIn(f' area {area} export-list {acl}', frrconfig)
 
 
     def test_ospf_14_segment_routing_configuration(self):
         global_block_low = "300"
         global_block_high = "399"
         local_block_low = "400"
         local_block_high = "499"
         interface = 'lo'
         maximum_stack_size = '5'
         prefix_one = '192.168.0.1/32'
         prefix_two = '192.168.0.2/32'
         prefix_one_value = '1'
         prefix_two_value = '2'
 
         self.cli_set(base_path + ['interface', interface])
         self.cli_set(base_path + ['segment-routing', 'maximum-label-depth', maximum_stack_size])
         self.cli_set(base_path + ['segment-routing', 'global-block', 'low-label-value', global_block_low])
         self.cli_set(base_path + ['segment-routing', 'global-block', 'high-label-value', global_block_high])
         self.cli_set(base_path + ['segment-routing', 'local-block', 'low-label-value', local_block_low])
         self.cli_set(base_path + ['segment-routing', 'local-block', 'high-label-value', local_block_high])
         self.cli_set(base_path + ['segment-routing', 'prefix', prefix_one, 'index', 'value', prefix_one_value])
         self.cli_set(base_path + ['segment-routing', 'prefix', prefix_one, 'index', 'explicit-null'])
         self.cli_set(base_path + ['segment-routing', 'prefix', prefix_two, 'index', 'value', prefix_two_value])
         self.cli_set(base_path + ['segment-routing', 'prefix', prefix_two, 'index', 'no-php-flag'])
 
         # Commit all changes
         self.cli_commit()
 
         # Verify all changes
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f' segment-routing on', frrconfig)
         self.assertIn(f' segment-routing global-block {global_block_low} {global_block_high} local-block {local_block_low} {local_block_high}', frrconfig)
         self.assertIn(f' segment-routing node-msd {maximum_stack_size}', frrconfig)
         self.assertIn(f' segment-routing prefix {prefix_one} index {prefix_one_value} explicit-null', frrconfig)
         self.assertIn(f' segment-routing prefix {prefix_two} index {prefix_two_value} no-php-flag', frrconfig)
 
     def test_ospf_15_ldp_sync(self):
         holddown = "500"
         interface = 'lo'
         interfaces = Section.interfaces('ethernet')
 
         self.cli_set(base_path + ['interface', interface])
         self.cli_set(base_path + ['ldp-sync', 'holddown', holddown])
 
         # Commit main OSPF changes
         self.cli_commit()
 
         # Verify main OSPF changes
         frrconfig = self.getFRRconfig('router ospf')
         self.assertIn(f'router ospf', frrconfig)
         self.assertIn(f' timers throttle spf 200 1000 10000', frrconfig)
         self.assertIn(f' mpls ldp-sync holddown {holddown}', frrconfig)
 
         for interface in interfaces:
             self.cli_set(base_path + ['interface', interface, 'ldp-sync', 'holddown', holddown])
 
             # Commit interface changes for holddown
             self.cli_commit()
 
             # Verify interface changes for holddown
             config = self.getFRRconfig(f'interface {interface}')
             self.assertIn(f'interface {interface}', config)
             self.assertIn(f' ip ospf dead-interval 40', config)
             self.assertIn(f' ip ospf mpls ldp-sync', config)
             self.assertIn(f' ip ospf mpls ldp-sync holddown {holddown}', config)
 
         for interface in interfaces:
             self.cli_set(base_path + ['interface', interface, 'ldp-sync', 'disable'])
 
             # Commit interface changes for disable
             self.cli_commit()
 
             # Verify interface changes for disable
             config = self.getFRRconfig(f'interface {interface}')
             self.assertIn(f'interface {interface}', config)
             self.assertIn(f' ip ospf dead-interval 40', config)
             self.assertIn(f' no ip ospf mpls ldp-sync', config)
 
+    def test_ospf_16_graceful_restart(self):
+        period = '300'
+        supported_grace_time = '400'
+        router_ids = ['192.0.2.1', '192.0.2.2']
+
+        self.cli_set(base_path + ['capability', 'opaque'])
+        self.cli_set(base_path + ['graceful-restart', 'grace-period', period])
+        self.cli_set(base_path + ['graceful-restart', 'helper', 'planned-only'])
+        self.cli_set(base_path + ['graceful-restart', 'helper', 'no-strict-lsa-checking'])
+        self.cli_set(base_path + ['graceful-restart', 'helper', 'supported-grace-time', supported_grace_time])
+        for router_id in router_ids:
+            self.cli_set(base_path + ['graceful-restart', 'helper', 'enable', 'router-id', router_id])
+
+        # commit changes
+        self.cli_commit()
+
+        # Verify FRR ospfd configuration
+        frrconfig = self.getFRRconfig('router ospf')
+        self.assertIn(f'router ospf', frrconfig)
+        self.assertIn(f' capability opaque', frrconfig)
+        self.assertIn(f' graceful-restart grace-period {period}', frrconfig)
+        self.assertIn(f' graceful-restart helper planned-only', frrconfig)
+        self.assertIn(f' no graceful-restart helper strict-lsa-checking', frrconfig)
+        self.assertIn(f' graceful-restart helper supported-grace-time {supported_grace_time}', frrconfig)
+        for router_id in router_ids:
+            self.assertIn(f' graceful-restart helper enable {router_id}', frrconfig)
+
 if __name__ == '__main__':
     unittest.main(verbosity=2)
diff --git a/smoketest/scripts/cli/test_protocols_ospfv3.py b/smoketest/scripts/cli/test_protocols_ospfv3.py
index b67bfaac7..64dfa18db 100755
--- a/smoketest/scripts/cli/test_protocols_ospfv3.py
+++ b/smoketest/scripts/cli/test_protocols_ospfv3.py
@@ -1,285 +1,311 @@
 #!/usr/bin/env python3
 #
 # Copyright (C) 2021-2022 VyOS maintainers and contributors
 #
 # This program is free software; you can redistribute it and/or modify
 # it under the terms of the GNU General Public License version 2 or later as
 # published by the Free Software Foundation.
 #
 # This program is distributed in the hope that it will be useful,
 # but WITHOUT ANY WARRANTY; without even the implied warranty of
 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 # GNU General Public License for more details.
 #
 # You should have received a copy of the GNU General Public License
 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
 
 import unittest
 
 from base_vyostest_shim import VyOSUnitTestSHIM
 
 from vyos.configsession import ConfigSessionError
 from vyos.ifconfig import Section
 from vyos.utils.process import process_named_running
 
 PROCESS_NAME = 'ospf6d'
 base_path = ['protocols', 'ospfv3']
 
 route_map = 'foo-bar-baz-0815'
 
 router_id = '192.0.2.1'
 default_area = '0'
 
 class TestProtocolsOSPFv3(VyOSUnitTestSHIM.TestCase):
     @classmethod
     def setUpClass(cls):
         super(TestProtocolsOSPFv3, cls).setUpClass()
 
         cls.cli_set(cls, ['policy', 'route-map', route_map, 'rule', '10', 'action', 'permit'])
         cls.cli_set(cls, ['policy', 'route-map', route_map, 'rule', '20', 'action', 'permit'])
 
         # ensure we can also run this test on a live system - so lets clean
         # out the current configuration :)
         cls.cli_delete(cls, base_path)
 
     @classmethod
     def tearDownClass(cls):
         cls.cli_delete(cls, ['policy', 'route-map', route_map])
         super(TestProtocolsOSPFv3, cls).tearDownClass()
 
     def tearDown(self):
         # Check for running process
         self.assertTrue(process_named_running(PROCESS_NAME))
         self.cli_delete(base_path)
         self.cli_commit()
 
     def test_ospfv3_01_basic(self):
         seq = '10'
         prefix = '2001:db8::/32'
         acl_name = 'foo-acl-100'
 
         self.cli_set(['policy', 'access-list6', acl_name, 'rule', seq, 'action', 'permit'])
         self.cli_set(['policy', 'access-list6', acl_name, 'rule', seq, 'source', 'any'])
 
         self.cli_set(base_path + ['parameters', 'router-id', router_id])
         self.cli_set(base_path + ['area', default_area, 'range', prefix, 'advertise'])
         self.cli_set(base_path + ['area', default_area, 'export-list', acl_name])
         self.cli_set(base_path + ['area', default_area, 'import-list', acl_name])
 
         interfaces = Section.interfaces('ethernet')
         for interface in interfaces:
             self.cli_set(base_path + ['interface', interface, 'area', default_area])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f'router ospf6', frrconfig)
         self.assertIn(f' area {default_area} range {prefix}', frrconfig)
         self.assertIn(f' ospf6 router-id {router_id}', frrconfig)
         self.assertIn(f' area {default_area} import-list {acl_name}', frrconfig)
         self.assertIn(f' area {default_area} export-list {acl_name}', frrconfig)
 
         for interface in interfaces:
             if_config = self.getFRRconfig(f'interface {interface}', daemon='ospf6d')
             self.assertIn(f'ipv6 ospf6 area {default_area}', if_config)
 
         self.cli_delete(['policy', 'access-list6', acl_name])
 
 
     def test_ospfv3_02_distance(self):
         dist_global = '200'
         dist_external = '110'
         dist_inter_area = '120'
         dist_intra_area = '130'
 
         self.cli_set(base_path + ['distance', 'global', dist_global])
         self.cli_set(base_path + ['distance', 'ospfv3', 'external', dist_external])
         self.cli_set(base_path + ['distance', 'ospfv3', 'inter-area', dist_inter_area])
         self.cli_set(base_path + ['distance', 'ospfv3', 'intra-area', dist_intra_area])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f'router ospf6', frrconfig)
         self.assertIn(f' distance {dist_global}', frrconfig)
         self.assertIn(f' distance ospf6 intra-area {dist_intra_area} inter-area {dist_inter_area} external {dist_external}', frrconfig)
 
 
     def test_ospfv3_03_redistribute(self):
         route_map = 'foo-bar'
         route_map_seq = '10'
         redistribute = ['bgp', 'connected', 'kernel', 'ripng', 'static']
 
         self.cli_set(['policy', 'route-map', route_map, 'rule', route_map_seq, 'action', 'permit'])
 
         for protocol in redistribute:
             self.cli_set(base_path + ['redistribute', protocol, 'route-map', route_map])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f'router ospf6', frrconfig)
         for protocol in redistribute:
             self.assertIn(f' redistribute {protocol} route-map {route_map}', frrconfig)
 
 
     def test_ospfv3_04_interfaces(self):
         bfd_profile = 'vyos-ipv6'
 
         self.cli_set(base_path + ['parameters', 'router-id', router_id])
         self.cli_set(base_path + ['area', default_area])
 
         cost = '100'
         priority = '10'
         interfaces = Section.interfaces('ethernet')
         for interface in interfaces:
             if_base = base_path + ['interface', interface]
             self.cli_set(if_base + ['bfd', 'profile', bfd_profile])
             self.cli_set(if_base + ['cost', cost])
             self.cli_set(if_base + ['instance-id', '0'])
             self.cli_set(if_base + ['mtu-ignore'])
             self.cli_set(if_base + ['network', 'point-to-point'])
             self.cli_set(if_base + ['passive'])
             self.cli_set(if_base + ['priority', priority])
             cost = str(int(cost) + 10)
             priority = str(int(priority) + 5)
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f'router ospf6', frrconfig)
 
         cost = '100'
         priority = '10'
         for interface in interfaces:
             if_config = self.getFRRconfig(f'interface {interface}', daemon='ospf6d')
             self.assertIn(f'interface {interface}', if_config)
             self.assertIn(f' ipv6 ospf6 bfd', if_config)
             self.assertIn(f' ipv6 ospf6 bfd profile {bfd_profile}', if_config)
             self.assertIn(f' ipv6 ospf6 cost {cost}', if_config)
             self.assertIn(f' ipv6 ospf6 mtu-ignore', if_config)
             self.assertIn(f' ipv6 ospf6 network point-to-point', if_config)
             self.assertIn(f' ipv6 ospf6 passive', if_config)
             self.assertIn(f' ipv6 ospf6 priority {priority}', if_config)
             cost = str(int(cost) + 10)
             priority = str(int(priority) + 5)
 
 
     def test_ospfv3_05_area_stub(self):
         area_stub = '23'
         area_stub_nosum = '26'
 
         self.cli_set(base_path + ['area', area_stub, 'area-type', 'stub'])
         self.cli_set(base_path + ['area', area_stub_nosum, 'area-type', 'stub', 'no-summary'])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f'router ospf6', frrconfig)
         self.assertIn(f' area {area_stub} stub', frrconfig)
         self.assertIn(f' area {area_stub_nosum} stub no-summary', frrconfig)
 
 
     def test_ospfv3_06_area_nssa(self):
         area_nssa = '1.1.1.1'
         area_nssa_nosum = '2.2.2.2'
         area_nssa_default = '3.3.3.3'
 
         self.cli_set(base_path + ['area', area_nssa, 'area-type', 'nssa'])
         self.cli_set(base_path + ['area', area_nssa, 'area-type', 'stub'])
         # can only set one area-type per OSPFv3 area
         with self.assertRaises(ConfigSessionError):
             self.cli_commit()
         self.cli_delete(base_path + ['area', area_nssa, 'area-type', 'stub'])
 
         self.cli_set(base_path + ['area', area_nssa_nosum, 'area-type', 'nssa', 'no-summary'])
         self.cli_set(base_path + ['area', area_nssa_nosum, 'area-type', 'nssa', 'default-information-originate'])
         self.cli_set(base_path + ['area', area_nssa_default, 'area-type', 'nssa', 'default-information-originate'])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f'router ospf6', frrconfig)
         self.assertIn(f' area {area_nssa} nssa', frrconfig)
         self.assertIn(f' area {area_nssa_nosum} nssa default-information-originate no-summary', frrconfig)
         self.assertIn(f' area {area_nssa_default} nssa default-information-originate', frrconfig)
 
 
     def test_ospfv3_07_default_originate(self):
         seq = '100'
         metric = '50'
         metric_type = '1'
 
         self.cli_set(base_path + ['default-information', 'originate', 'metric', metric])
         self.cli_set(base_path + ['default-information', 'originate', 'metric-type', metric_type])
         self.cli_set(base_path + ['default-information', 'originate', 'route-map', route_map])
 
         # commit changes
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f'router ospf6', frrconfig)
         self.assertIn(f' default-information originate metric {metric} metric-type {metric_type} route-map {route_map}', frrconfig)
 
         # Now set 'always'
         self.cli_set(base_path + ['default-information', 'originate', 'always'])
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f' default-information originate always metric {metric} metric-type {metric_type} route-map {route_map}', frrconfig)
 
 
     def test_ospfv3_08_vrfs(self):
         # It is safe to assume that when the basic VRF test works, all
         # other OSPF related features work, as we entirely inherit the CLI
         # templates and Jinja2 FRR template.
         table = '1000'
         vrf = 'blue'
         vrf_base = ['vrf', 'name', vrf]
         vrf_iface = 'eth1'
         router_id = '1.2.3.4'
         router_id_vrf = '1.2.3.5'
 
         self.cli_set(vrf_base + ['table', table])
         self.cli_set(vrf_base + ['protocols', 'ospfv3', 'interface', vrf_iface, 'bfd'])
         self.cli_set(vrf_base + ['protocols', 'ospfv3', 'parameters', 'router-id', router_id_vrf])
 
         self.cli_set(['interfaces', 'ethernet', vrf_iface, 'vrf', vrf])
 
         # Also set a default VRF OSPF config
         self.cli_set(base_path + ['parameters', 'router-id', router_id])
         self.cli_commit()
 
         # Verify FRR ospfd configuration
         frrconfig = self.getFRRconfig('router ospf6', daemon='ospf6d')
         self.assertIn(f'router ospf6', frrconfig)
         self.assertIn(f' ospf6 router-id {router_id}', frrconfig)
 
         frrconfig = self.getFRRconfig(f'interface {vrf_iface}', daemon='ospf6d')
         self.assertIn(f'interface {vrf_iface}', frrconfig)
         self.assertIn(f' ipv6 ospf6 bfd', frrconfig)
 
         frrconfig = self.getFRRconfig(f'router ospf6 vrf {vrf}', daemon='ospf6d')
         self.assertIn(f'router ospf6 vrf {vrf}', frrconfig)
         self.assertIn(f' ospf6 router-id {router_id_vrf}', frrconfig)
 
         # cleanup
         self.cli_delete(['vrf', 'name', vrf])
         self.cli_delete(['interfaces', 'ethernet', vrf_iface, 'vrf'])
 
+
+    def test_ospfv3_09_graceful_restart(self):
+        period = '300'
+        supported_grace_time = '400'
+        router_ids = ['192.0.2.1', '192.0.2.2']
+
+        self.cli_set(base_path + ['graceful-restart', 'grace-period', period])
+        self.cli_set(base_path + ['graceful-restart', 'helper', 'planned-only'])
+        self.cli_set(base_path + ['graceful-restart', 'helper', 'lsa-check-disable'])
+        self.cli_set(base_path + ['graceful-restart', 'helper', 'supported-grace-time', supported_grace_time])
+        for router_id in router_ids:
+            self.cli_set(base_path + ['graceful-restart', 'helper', 'enable', 'router-id', router_id])
+
+        # commit changes
+        self.cli_commit()
+
+        # Verify FRR ospfd configuration
+        frrconfig = self.getFRRconfig('router ospf6')
+        self.assertIn(f'router ospf6', frrconfig)
+        self.assertIn(f' graceful-restart grace-period {period}', frrconfig)
+        self.assertIn(f' graceful-restart helper planned-only', frrconfig)
+        self.assertIn(f' graceful-restart helper lsa-check-disable', frrconfig)
+        self.assertIn(f' graceful-restart helper supported-grace-time {supported_grace_time}', frrconfig)
+        for router_id in router_ids:
+            self.assertIn(f' graceful-restart helper enable {router_id}', frrconfig)
+
 if __name__ == '__main__':
     unittest.main(verbosity=2)
diff --git a/src/conf_mode/protocols_ospf.py b/src/conf_mode/protocols_ospf.py
index 509d4f501..f2075d25b 100755
--- a/src/conf_mode/protocols_ospf.py
+++ b/src/conf_mode/protocols_ospf.py
@@ -1,305 +1,307 @@
 #!/usr/bin/env python3
 #
 # Copyright (C) 2021 VyOS maintainers and contributors
 #
 # This program is free software; you can redistribute it and/or modify
 # it under the terms of the GNU General Public License version 2 or later as
 # published by the Free Software Foundation.
 #
 # This program is distributed in the hope that it will be useful,
 # but WITHOUT ANY WARRANTY; without even the implied warranty of
 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 # GNU General Public License for more details.
 #
 # You should have received a copy of the GNU General Public License
 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
 
 import os
 
 from sys import exit
 from sys import argv
 
 from vyos.config import Config
 from vyos.configdict import dict_merge
 from vyos.configdict import node_changed
 from vyos.configverify import verify_common_route_maps
 from vyos.configverify import verify_route_map
 from vyos.configverify import verify_interface_exists
 from vyos.configverify import verify_access_list
 from vyos.template import render_to_string
 from vyos.utils.dict import dict_search
 from vyos.utils.network import get_interface_config
 from vyos.xml import defaults
 from vyos import ConfigError
 from vyos import frr
 from vyos import airbag
 airbag.enable()
 
 def get_config(config=None):
     if config:
         conf = config
     else:
         conf = Config()
 
     vrf = None
     if len(argv) > 1:
         vrf = argv[1]
 
     base_path = ['protocols', 'ospf']
 
     # eqivalent of the C foo ? 'a' : 'b' statement
     base = vrf and ['vrf', 'name', vrf, 'protocols', 'ospf'] or base_path
     ospf = conf.get_config_dict(base, key_mangling=('-', '_'),
                                 get_first_key=True)
 
     # Assign the name of our VRF context. This MUST be done before the return
     # statement below, else on deletion we will delete the default instance
     # instead of the VRF instance.
     if vrf: ospf['vrf'] = vrf
 
     # FRR has VRF support for different routing daemons. As interfaces belong
     # to VRFs - or the global VRF, we need to check for changed interfaces so
     # that they will be properly rendered for the FRR config. Also this eases
     # removal of interfaces from the running configuration.
     interfaces_removed = node_changed(conf, base + ['interface'])
     if interfaces_removed:
         ospf['interface_removed'] = list(interfaces_removed)
 
     # Bail out early if configuration tree does not exist
     if not conf.exists(base):
         ospf.update({'deleted' : ''})
         return ospf
 
     # We have gathered the dict representation of the CLI, but there are default
     # options which we need to update into the dictionary retrived.
     # XXX: Note that we can not call defaults(base), as defaults does not work
     # on an instance of a tag node. As we use the exact same CLI definition for
     # both the non-vrf and vrf version this is absolutely safe!
     default_values = defaults(base_path)
 
     # We have to cleanup the default dict, as default values could enable features
     # which are not explicitly enabled on the CLI. Example: default-information
     # originate comes with a default metric-type of 2, which will enable the
     # entire default-information originate tree, even when not set via CLI so we
     # need to check this first and probably drop that key.
     if dict_search('default_information.originate', ospf) is None:
         del default_values['default_information']
     if dict_search('area.area_type.nssa', ospf) is None:
         del default_values['area']['area_type']['nssa']
     if 'mpls_te' not in ospf:
         del default_values['mpls_te']
+    if 'graceful_restart' not in ospf:
+        del default_values['graceful_restart']
 
     for protocol in ['babel', 'bgp', 'connected', 'isis', 'kernel', 'rip', 'static', 'table']:
         # table is a tagNode thus we need to clean out all occurances for the
         # default values and load them in later individually
         if protocol == 'table':
             del default_values['redistribute']['table']
             continue
         if dict_search(f'redistribute.{protocol}', ospf) is None:
             del default_values['redistribute'][protocol]
 
     # XXX: T2665: we currently have no nice way for defaults under tag nodes,
     # clean them out and add them manually :(
     del default_values['neighbor']
     del default_values['area']['virtual_link']
     del default_values['interface']
 
     # merge in remaining default values
     ospf = dict_merge(default_values, ospf)
 
     if 'neighbor' in ospf:
         default_values = defaults(base + ['neighbor'])
         for neighbor in ospf['neighbor']:
             ospf['neighbor'][neighbor] = dict_merge(default_values, ospf['neighbor'][neighbor])
 
     if 'area' in ospf:
         default_values = defaults(base + ['area', 'virtual-link'])
         for area, area_config in ospf['area'].items():
             if 'virtual_link' in area_config:
                 for virtual_link in area_config['virtual_link']:
                     ospf['area'][area]['virtual_link'][virtual_link] = dict_merge(
                         default_values, ospf['area'][area]['virtual_link'][virtual_link])
 
     if 'interface' in ospf:
         for interface in ospf['interface']:
             # We need to reload the defaults on every pass b/c of
             # hello-multiplier dependency on dead-interval
             default_values = defaults(base + ['interface'])
             # If hello-multiplier is set, we need to remove the default from
             # dead-interval.
             if 'hello_multiplier' in ospf['interface'][interface]:
                 del default_values['dead_interval']
 
             ospf['interface'][interface] = dict_merge(default_values,
                 ospf['interface'][interface])
 
     if 'redistribute' in ospf and 'table' in ospf['redistribute']:
         default_values = defaults(base + ['redistribute', 'table'])
         for table in ospf['redistribute']['table']:
             ospf['redistribute']['table'][table] = dict_merge(default_values,
                 ospf['redistribute']['table'][table])
 
     # We also need some additional information from the config, prefix-lists
     # and route-maps for instance. They will be used in verify().
     #
     # XXX: one MUST always call this without the key_mangling() option! See
     # vyos.configverify.verify_common_route_maps() for more information.
     tmp = conf.get_config_dict(['policy'])
     # Merge policy dict into "regular" config dict
     ospf = dict_merge(tmp, ospf)
 
     return ospf
 
 def verify(ospf):
     if not ospf:
         return None
 
     verify_common_route_maps(ospf)
 
     # As we can have a default-information route-map, we need to validate it!
     route_map_name = dict_search('default_information.originate.route_map', ospf)
     if route_map_name: verify_route_map(route_map_name, ospf)
 
     # Validate if configured Access-list exists
     if 'area' in ospf:
           for area, area_config in ospf['area'].items():
               if 'import_list' in area_config:
                   acl_import = area_config['import_list']
                   if acl_import: verify_access_list(acl_import, ospf)
               if 'export_list' in area_config:
                   acl_export = area_config['export_list']
                   if acl_export: verify_access_list(acl_export, ospf)
 
     if 'interface' in ospf:
         for interface, interface_config in ospf['interface'].items():
             verify_interface_exists(interface)
             # One can not use dead-interval and hello-multiplier at the same
             # time. FRR will only activate the last option set via CLI.
             if {'hello_multiplier', 'dead_interval'} <= set(interface_config):
                 raise ConfigError(f'Can not use hello-multiplier and dead-interval ' \
                                   f'concurrently for {interface}!')
 
             # One can not use the "network <prefix> area <id>" command and an
             # per interface area assignment at the same time. FRR will error
             # out using: "Please remove all network commands first."
             if 'area' in ospf and 'area' in interface_config:
                 for area, area_config in ospf['area'].items():
                     if 'network' in area_config:
                         raise ConfigError('Can not use OSPF interface area and area ' \
                                           'network configuration at the same time!')
 
             # If interface specific options are set, we must ensure that the
             # interface is bound to our requesting VRF. Due to the VyOS
             # priorities the interface is bound to the VRF after creation of
             # the VRF itself, and before any routing protocol is configured.
             if 'vrf' in ospf:
                 vrf = ospf['vrf']
                 tmp = get_interface_config(interface)
                 if 'master' not in tmp or tmp['master'] != vrf:
                     raise ConfigError(f'Interface "{interface}" is not a member of VRF "{vrf}"!')
 
     # Segment routing checks
     if dict_search('segment_routing.global_block', ospf):
         g_high_label_value = dict_search('segment_routing.global_block.high_label_value', ospf)
         g_low_label_value = dict_search('segment_routing.global_block.low_label_value', ospf)
 
         # If segment routing global block high or low value is blank, throw error
         if not (g_low_label_value or g_high_label_value):
             raise ConfigError('Segment routing global-block requires both low and high value!')
 
         # If segment routing global block low value is higher than the high value, throw error
         if int(g_low_label_value) > int(g_high_label_value):
             raise ConfigError('Segment routing global-block low value must be lower than high value')
 
     if dict_search('segment_routing.local_block', ospf):
         if dict_search('segment_routing.global_block', ospf) == None:
             raise ConfigError('Segment routing local-block requires global-block to be configured!')
 
         l_high_label_value = dict_search('segment_routing.local_block.high_label_value', ospf)
         l_low_label_value = dict_search('segment_routing.local_block.low_label_value', ospf)
 
         # If segment routing local-block high or low value is blank, throw error
         if not (l_low_label_value or l_high_label_value):
             raise ConfigError('Segment routing local-block requires both high and low value!')
 
         # If segment routing local-block low value is higher than the high value, throw error
         if int(l_low_label_value) > int(l_high_label_value):
             raise ConfigError('Segment routing local-block low value must be lower than high value')
 
         # local-block most live outside global block
         global_range = range(int(g_low_label_value), int(g_high_label_value) +1)
         local_range  = range(int(l_low_label_value), int(l_high_label_value) +1)
 
         # Check for overlapping ranges
         if list(set(global_range) & set(local_range)):
             raise ConfigError(f'Segment-Routing Global Block ({g_low_label_value}/{g_high_label_value}) '\
                               f'conflicts with Local Block ({l_low_label_value}/{l_high_label_value})!')
 
     # Check for a blank or invalid value per prefix
     if dict_search('segment_routing.prefix', ospf):
         for prefix, prefix_config in ospf['segment_routing']['prefix'].items():
             if 'index' in prefix_config:
                 if prefix_config['index'].get('value') is None:
                     raise ConfigError(f'Segment routing prefix {prefix} index value cannot be blank.')
 
     # Check for explicit-null and no-php-flag configured at the same time per prefix
     if dict_search('segment_routing.prefix', ospf):
         for prefix, prefix_config in ospf['segment_routing']['prefix'].items():
             if 'index' in prefix_config:
                 if ("explicit_null" in prefix_config['index']) and ("no_php_flag" in prefix_config['index']):
                     raise ConfigError(f'Segment routing prefix {prefix} cannot have both explicit-null '\
                                       f'and no-php-flag configured at the same time.')
 
     # Check route summarisation
     if 'summary_address' in ospf:
         for prefix, prefix_options in ospf['summary_address'].items():
             if {'tag', 'no_advertise'} <= set(prefix_options):
                 raise ConfigError(f'Can not set both "tag" and "no-advertise" for Type-5 '\
                                   f'and Type-7 route summarisation of "{prefix}"!')
 
     return None
 
 def generate(ospf):
     if not ospf or 'deleted' in ospf:
         return None
 
     ospf['frr_ospfd_config'] = render_to_string('frr/ospfd.frr.j2', ospf)
     return None
 
 def apply(ospf):
     ospf_daemon = 'ospfd'
 
     # Save original configuration prior to starting any commit actions
     frr_cfg = frr.FRRConfig()
 
     # Generate empty helper string which can be ammended to FRR commands, it
     # will be either empty (default VRF) or contain the "vrf <name" statement
     vrf = ''
     if 'vrf' in ospf:
         vrf = ' vrf ' + ospf['vrf']
 
     frr_cfg.load_configuration(ospf_daemon)
     frr_cfg.modify_section(f'^router ospf{vrf}', stop_pattern='^exit', remove_stop_mark=True)
 
     for key in ['interface', 'interface_removed']:
         if key not in ospf:
             continue
         for interface in ospf[key]:
             frr_cfg.modify_section(f'^interface {interface}{vrf}', stop_pattern='^exit', remove_stop_mark=True)
 
     if 'frr_ospfd_config' in ospf:
         frr_cfg.add_before(frr.default_add_before, ospf['frr_ospfd_config'])
 
     frr_cfg.commit_configuration(ospf_daemon)
 
     return None
 
 if __name__ == '__main__':
     try:
         c = get_config()
         verify(c)
         generate(c)
         apply(c)
     except ConfigError as e:
         print(e)
         exit(1)
diff --git a/src/conf_mode/protocols_ospfv3.py b/src/conf_mode/protocols_ospfv3.py
index 7f50d8624..fbea51f56 100755
--- a/src/conf_mode/protocols_ospfv3.py
+++ b/src/conf_mode/protocols_ospfv3.py
@@ -1,188 +1,190 @@
 #!/usr/bin/env python3
 #
 # Copyright (C) 2021-2023 VyOS maintainers and contributors
 #
 # This program is free software; you can redistribute it and/or modify
 # it under the terms of the GNU General Public License version 2 or later as
 # published by the Free Software Foundation.
 #
 # This program is distributed in the hope that it will be useful,
 # but WITHOUT ANY WARRANTY; without even the implied warranty of
 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 # GNU General Public License for more details.
 #
 # You should have received a copy of the GNU General Public License
 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
 
 import os
 
 from sys import exit
 from sys import argv
 
 from vyos.config import Config
 from vyos.configdict import dict_merge
 from vyos.configdict import node_changed
 from vyos.configverify import verify_common_route_maps
 from vyos.configverify import verify_route_map
 from vyos.configverify import verify_interface_exists
 from vyos.template import render_to_string
 from vyos.ifconfig import Interface
 from vyos.utils.dict import dict_search
 from vyos.utils.network import get_interface_config
 from vyos.xml import defaults
 from vyos import ConfigError
 from vyos import frr
 from vyos import airbag
 airbag.enable()
 
 def get_config(config=None):
     if config:
         conf = config
     else:
         conf = Config()
 
     vrf = None
     if len(argv) > 1:
         vrf = argv[1]
 
     base_path = ['protocols', 'ospfv3']
 
     # eqivalent of the C foo ? 'a' : 'b' statement
     base = vrf and ['vrf', 'name', vrf, 'protocols', 'ospfv3'] or base_path
     ospfv3 = conf.get_config_dict(base, key_mangling=('-', '_'), get_first_key=True)
 
     # Assign the name of our VRF context. This MUST be done before the return
     # statement below, else on deletion we will delete the default instance
     # instead of the VRF instance.
     if vrf: ospfv3['vrf'] = vrf
 
     # FRR has VRF support for different routing daemons. As interfaces belong
     # to VRFs - or the global VRF, we need to check for changed interfaces so
     # that they will be properly rendered for the FRR config. Also this eases
     # removal of interfaces from the running configuration.
     interfaces_removed = node_changed(conf, base + ['interface'])
     if interfaces_removed:
         ospfv3['interface_removed'] = list(interfaces_removed)
 
     # Bail out early if configuration tree does not exist
     if not conf.exists(base):
         ospfv3.update({'deleted' : ''})
         return ospfv3
 
     # We have gathered the dict representation of the CLI, but there are default
     # options which we need to update into the dictionary retrived.
     # XXX: Note that we can not call defaults(base), as defaults does not work
     # on an instance of a tag node. As we use the exact same CLI definition for
     # both the non-vrf and vrf version this is absolutely safe!
     default_values = defaults(base_path)
 
     # We have to cleanup the default dict, as default values could enable features
     # which are not explicitly enabled on the CLI. Example: default-information
     # originate comes with a default metric-type of 2, which will enable the
     # entire default-information originate tree, even when not set via CLI so we
     # need to check this first and probably drop that key.
     if dict_search('default_information.originate', ospfv3) is None:
         del default_values['default_information']
+    if 'graceful_restart' not in ospfv3:
+        del default_values['graceful_restart']
 
     # XXX: T2665: we currently have no nice way for defaults under tag nodes,
     # clean them out and add them manually :(
     del default_values['interface']
 
     # merge in remaining default values
     ospfv3 = dict_merge(default_values, ospfv3)
 
     # We also need some additional information from the config, prefix-lists
     # and route-maps for instance. They will be used in verify().
     #
     # XXX: one MUST always call this without the key_mangling() option! See
     # vyos.configverify.verify_common_route_maps() for more information.
     tmp = conf.get_config_dict(['policy'])
     # Merge policy dict into "regular" config dict
     ospfv3 = dict_merge(tmp, ospfv3)
 
     return ospfv3
 
 def verify(ospfv3):
     if not ospfv3:
         return None
 
     verify_common_route_maps(ospfv3)
 
     # As we can have a default-information route-map, we need to validate it!
     route_map_name = dict_search('default_information.originate.route_map', ospfv3)
     if route_map_name: verify_route_map(route_map_name, ospfv3)
 
     if 'area' in ospfv3:
         for area, area_config in ospfv3['area'].items():
             if 'area_type' in area_config:
                 if len(area_config['area_type']) > 1:
                     raise ConfigError(f'Can only configure one area-type for OSPFv3 area "{area}"!')
             if 'range' in area_config:
                 for range, range_config in area_config['range'].items():
                     if {'not_advertise', 'advertise'} <= range_config.keys():
                         raise ConfigError(f'"not-advertise" and "advertise" for "range {range}" cannot be both configured at the same time!')
 
     if 'interface' in ospfv3:
         for interface, interface_config in ospfv3['interface'].items():
             verify_interface_exists(interface)
             if 'ifmtu' in interface_config:
                 mtu = Interface(interface).get_mtu()
                 if int(interface_config['ifmtu']) > int(mtu):
                     raise ConfigError(f'OSPFv3 ifmtu can not exceed physical MTU of "{mtu}"')
 
             # If interface specific options are set, we must ensure that the
             # interface is bound to our requesting VRF. Due to the VyOS
             # priorities the interface is bound to the VRF after creation of
             # the VRF itself, and before any routing protocol is configured.
             if 'vrf' in ospfv3:
                 vrf = ospfv3['vrf']
                 tmp = get_interface_config(interface)
                 if 'master' not in tmp or tmp['master'] != vrf:
                     raise ConfigError(f'Interface "{interface}" is not a member of VRF "{vrf}"!')
 
     return None
 
 def generate(ospfv3):
     if not ospfv3 or 'deleted' in ospfv3:
         return None
 
     ospfv3['new_frr_config'] = render_to_string('frr/ospf6d.frr.j2', ospfv3)
     return None
 
 def apply(ospfv3):
     ospf6_daemon = 'ospf6d'
 
     # Save original configuration prior to starting any commit actions
     frr_cfg = frr.FRRConfig()
 
     # Generate empty helper string which can be ammended to FRR commands, it
     # will be either empty (default VRF) or contain the "vrf <name" statement
     vrf = ''
     if 'vrf' in ospfv3:
         vrf = ' vrf ' + ospfv3['vrf']
 
     frr_cfg.load_configuration(ospf6_daemon)
     frr_cfg.modify_section(f'^router ospf6{vrf}', stop_pattern='^exit', remove_stop_mark=True)
 
     for key in ['interface', 'interface_removed']:
         if key not in ospfv3:
             continue
         for interface in ospfv3[key]:
             frr_cfg.modify_section(f'^interface {interface}{vrf}', stop_pattern='^exit', remove_stop_mark=True)
 
     if 'new_frr_config' in ospfv3:
         frr_cfg.add_before(frr.default_add_before, ospfv3['new_frr_config'])
 
     frr_cfg.commit_configuration(ospf6_daemon)
 
     return None
 
 if __name__ == '__main__':
     try:
         c = get_config()
         verify(c)
         generate(c)
         apply(c)
     except ConfigError as e:
         print(e)
         exit(1)