diff --git a/interface-definitions/include/qos/class-match-group.xml.i b/interface-definitions/include/qos/class-match-group.xml.i
new file mode 100644
index 000000000..40e3b7259
--- /dev/null
+++ b/interface-definitions/include/qos/class-match-group.xml.i
@@ -0,0 +1,15 @@
+<!-- include start from qos/class-match-group.xml.i -->
+<leafNode name="match-group">
+  <properties>
+    <help>Filter group for QoS policy</help>
+    <valueHelp>
+      <format>txt</format>
+      <description>Match group name</description>
+    </valueHelp>
+    <completionHelp>
+      <script>${vyos_completion_dir}/qos/list_traffic_match_group.py</script>
+    </completionHelp>
+    <multi/>
+  </properties>
+</leafNode>
+<!-- include end -->
diff --git a/interface-definitions/include/qos/class-match-ipv4.xml.i b/interface-definitions/include/qos/class-match-ipv4.xml.i
new file mode 100644
index 000000000..dc44d32d5
--- /dev/null
+++ b/interface-definitions/include/qos/class-match-ipv4.xml.i
@@ -0,0 +1,31 @@
+<!-- include start from qos/class-match-ipv4.xml.i -->
+<node name="ip">
+  <properties>
+    <help>Match IP protocol header</help>
+  </properties>
+  <children>
+    <node name="destination">
+      <properties>
+        <help>Match on destination port or address</help>
+      </properties>
+      <children>
+        #include <include/qos/class-match-ipv4-address.xml.i>
+        #include <include/port-number.xml.i>
+      </children>
+    </node>
+    #include <include/qos/match-dscp.xml.i>
+    #include <include/qos/max-length.xml.i>
+    #include <include/ip-protocol.xml.i>
+    <node name="source">
+      <properties>
+        <help>Match on source port or address</help>
+      </properties>
+      <children>
+        #include <include/qos/class-match-ipv4-address.xml.i>
+        #include <include/port-number.xml.i>
+      </children>
+    </node>
+    #include <include/qos/tcp-flags.xml.i>
+  </children>
+</node>
+<!-- include end -->
diff --git a/interface-definitions/include/qos/class-match-ipv6.xml.i b/interface-definitions/include/qos/class-match-ipv6.xml.i
new file mode 100644
index 000000000..ed7aceff9
--- /dev/null
+++ b/interface-definitions/include/qos/class-match-ipv6.xml.i
@@ -0,0 +1,31 @@
+<!-- include start from qos/class-match-ipv6.xml.i -->
+<node name="ipv6">
+  <properties>
+    <help>Match IPv6 protocol header</help>
+  </properties>
+  <children>
+    <node name="destination">
+      <properties>
+        <help>Match on destination port or address</help>
+      </properties>
+      <children>
+        #include <include/qos/class-match-ipv6-address.xml.i>
+        #include <include/port-number.xml.i>
+      </children>
+    </node>
+    #include <include/qos/match-dscp.xml.i>
+    #include <include/qos/max-length.xml.i>
+    #include <include/ip-protocol.xml.i>
+    <node name="source">
+      <properties>
+        <help>Match on source port or address</help>
+      </properties>
+      <children>
+        #include <include/qos/class-match-ipv6-address.xml.i>
+        #include <include/port-number.xml.i>
+      </children>
+    </node>
+    #include <include/qos/tcp-flags.xml.i>
+  </children>
+</node>
+<!-- include end -->
diff --git a/interface-definitions/include/qos/class-match-mark.xml.i b/interface-definitions/include/qos/class-match-mark.xml.i
new file mode 100644
index 000000000..a7481c6aa
--- /dev/null
+++ b/interface-definitions/include/qos/class-match-mark.xml.i
@@ -0,0 +1,14 @@
+<!-- include start from qos/class-match-mark.xml.i -->
+<leafNode name="mark">
+  <properties>
+    <help>Match on mark applied by firewall</help>
+    <valueHelp>
+      <format>u32</format>
+      <description>FW mark to match</description>
+    </valueHelp>
+    <constraint>
+      <validator name="numeric" argument="--range 0-4294967295"/>
+    </constraint>
+  </properties>
+</leafNode>
+<!-- include end -->
diff --git a/interface-definitions/include/qos/class-match-vif.xml.i b/interface-definitions/include/qos/class-match-vif.xml.i
new file mode 100644
index 000000000..ec58db606
--- /dev/null
+++ b/interface-definitions/include/qos/class-match-vif.xml.i
@@ -0,0 +1,15 @@
+<!-- include start from qos/class-match-vif.xml.i -->
+<leafNode name="vif">
+  <properties>
+    <help>Virtual Local Area Network (VLAN) ID for this match</help>
+    <valueHelp>
+      <format>u32:0-4095</format>
+      <description>Virtual Local Area Network (VLAN) tag </description>
+    </valueHelp>
+    <constraint>
+      <validator name="numeric" argument="--range 0-4095"/>
+    </constraint>
+    <constraintErrorMessage>VLAN ID must be between 0 and 4095</constraintErrorMessage>
+  </properties>
+</leafNode>
+<!-- include end -->
diff --git a/interface-definitions/include/qos/class-match.xml.i b/interface-definitions/include/qos/class-match.xml.i
index 4ba12f8f7..77d1933a3 100644
--- a/interface-definitions/include/qos/class-match.xml.i
+++ b/interface-definitions/include/qos/class-match.xml.i
@@ -1,177 +1,98 @@
 <!-- include start from qos/class-match.xml.i -->
 <tagNode name="match">
   <properties>
     <help>Class matching rule name</help>
     <constraint>
       <regex>[^-].*</regex>
     </constraint>
-    <constraintErrorMessage>Match queue name cannot start with hyphen (-)</constraintErrorMessage>
+    <constraintErrorMessage>Match queue name cannot start with hyphen</constraintErrorMessage>
   </properties>
   <children>
     #include <include/generic-description.xml.i>
     <node name="ether">
       <properties>
         <help>Ethernet header match</help>
       </properties>
       <children>
         <leafNode name="destination">
           <properties>
             <help>Ethernet destination address for this match</help>
             <valueHelp>
               <format>macaddr</format>
               <description>MAC address to match</description>
             </valueHelp>
             <constraint>
               <validator name="mac-address"/>
             </constraint>
           </properties>
         </leafNode>
         <leafNode name="protocol">
           <properties>
             <help>Ethernet protocol for this match</help>
             <!-- this refers to /etc/protocols -->
             <completionHelp>
               <list>all 802.1Q 802_2 802_3 aarp aoe arp atalk dec ip ipv6 ipx lat localtalk rarp snap x25</list>
             </completionHelp>
             <valueHelp>
               <format>u32:0-65535</format>
               <description>Ethernet protocol number</description>
             </valueHelp>
             <valueHelp>
               <format>txt</format>
               <description>Ethernet protocol name</description>
             </valueHelp>
             <valueHelp>
               <format>all</format>
               <description>Any protocol</description>
             </valueHelp>
             <valueHelp>
               <format>ip</format>
               <description>Internet IP (IPv4)</description>
             </valueHelp>
             <valueHelp>
               <format>ipv6</format>
               <description>Internet IP (IPv6)</description>
             </valueHelp>
             <valueHelp>
               <format>arp</format>
               <description>Address Resolution Protocol</description>
             </valueHelp>
             <valueHelp>
               <format>atalk</format>
               <description>Appletalk</description>
             </valueHelp>
             <valueHelp>
               <format>ipx</format>
               <description>Novell Internet Packet Exchange</description>
             </valueHelp>
             <valueHelp>
               <format>802.1Q</format>
               <description>802.1Q VLAN tag</description>
             </valueHelp>
             <constraint>
               <validator name="ip-protocol"/>
             </constraint>
           </properties>
         </leafNode>
         <leafNode name="source">
           <properties>
             <help>Ethernet source address for this match</help>
             <valueHelp>
               <format>macaddr</format>
               <description>MAC address to match</description>
             </valueHelp>
             <constraint>
               <validator name="mac-address"/>
             </constraint>
           </properties>
         </leafNode>
       </children>
     </node>
     #include <include/generic-interface.xml.i>
-    <node name="ip">
-      <properties>
-        <help>Match IP protocol header</help>
-      </properties>
-      <children>
-        <node name="destination">
-          <properties>
-            <help>Match on destination port or address</help>
-          </properties>
-          <children>
-            #include <include/qos/class-match-ipv4-address.xml.i>
-            #include <include/port-number.xml.i>
-          </children>
-        </node>
-        #include <include/qos/match-dscp.xml.i>
-        #include <include/qos/max-length.xml.i>
-        #include <include/ip-protocol.xml.i>
-        <node name="source">
-          <properties>
-            <help>Match on source port or address</help>
-          </properties>
-          <children>
-            #include <include/qos/class-match-ipv4-address.xml.i>
-            #include <include/port-number.xml.i>
-          </children>
-        </node>
-        #include <include/qos/tcp-flags.xml.i>
-      </children>
-    </node>
-    <node name="ipv6">
-      <properties>
-        <help>Match IPv6 protocol header</help>
-      </properties>
-      <children>
-        <node name="destination">
-          <properties>
-            <help>Match on destination port or address</help>
-          </properties>
-          <children>
-            #include <include/qos/class-match-ipv6-address.xml.i>
-            #include <include/port-number.xml.i>
-          </children>
-        </node>
-        #include <include/qos/match-dscp.xml.i>
-        #include <include/qos/max-length.xml.i>
-        #include <include/ip-protocol.xml.i>
-        <node name="source">
-          <properties>
-            <help>Match on source port or address</help>
-          </properties>
-          <children>
-            #include <include/qos/class-match-ipv6-address.xml.i>
-            #include <include/port-number.xml.i>
-          </children>
-        </node>
-        #include <include/qos/tcp-flags.xml.i>
-      </children>
-    </node>
-    <leafNode name="mark">
-      <properties>
-        <help>Match on mark applied by firewall</help>
-        <valueHelp>
-          <format>u32</format>
-          <description>FW mark to match</description>
-        </valueHelp>
-        <constraint>
-          <validator name="numeric" argument="--range 0-4294967295"/>
-        </constraint>
-      </properties>
-    </leafNode>
-    <leafNode name="vif">
-      <properties>
-        <help>Virtual Local Area Network (VLAN) ID for this match</help>
-        <valueHelp>
-          <format>u32:0-4095</format>
-          <description>Virtual Local Area Network (VLAN) tag </description>
-        </valueHelp>
-        <constraint>
-          <validator name="numeric" argument="--range 0-4095"/>
-        </constraint>
-        <constraintErrorMessage>VLAN ID must be between 0 and 4095</constraintErrorMessage>
-      </properties>
-    </leafNode>
+    #include <include/qos/class-match-ipv4.xml.i>
+    #include <include/qos/class-match-ipv6.xml.i>
+    #include <include/qos/class-match-mark.xml.i>
+    #include <include/qos/class-match-vif.xml.i>
   </children>
 </tagNode>
 <!-- include end -->
diff --git a/interface-definitions/qos.xml.in b/interface-definitions/qos.xml.in
index 8f9ae3fa6..927594c11 100644
--- a/interface-definitions/qos.xml.in
+++ b/interface-definitions/qos.xml.in
@@ -1,835 +1,874 @@
 <?xml version="1.0"?>
 <interfaceDefinition>
   <node name="qos" owner="${vyos_conf_scripts_dir}/qos.py">
     <properties>
       <help>Quality of Service (QoS)</help>
       <priority>900</priority>
     </properties>
     <children>
       <tagNode name="interface">
         <properties>
           <help>Interface to apply QoS policy</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="ingress">
             <properties>
               <help>Interface ingress traffic policy</help>
               <completionHelp>
                 <path>qos policy limiter</path>
               </completionHelp>
               <valueHelp>
                 <format>txt</format>
                 <description>QoS policy to use</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
           </leafNode>
           <leafNode name="egress">
             <properties>
               <help>Interface egress traffic policy</help>
               <completionHelp>
                 <path>qos policy cake</path>
                 <path>qos policy drop-tail</path>
                 <path>qos policy fair-queue</path>
                 <path>qos policy fq-codel</path>
                 <path>qos policy network-emulator</path>
                 <path>qos policy priority-queue</path>
                 <path>qos policy random-detect</path>
                 <path>qos policy rate-control</path>
                 <path>qos policy round-robin</path>
                 <path>qos policy shaper</path>
                 <path>qos policy shaper-hfsc</path>
               </completionHelp>
               <valueHelp>
                 <format>txt</format>
                 <description>QoS policy to use</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
           </leafNode>
         </children>
       </tagNode>
       <node name="policy">
         <properties>
           <help>Service Policy definitions</help>
         </properties>
         <children>
           <tagNode name="cake">
             <properties>
               <help>Common Applications Kept Enhanced (CAKE)</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               #include <include/qos/bandwidth.xml.i>
               <node name="flow-isolation">
                 <properties>
                   <help>Flow isolation settings</help>
                 </properties>
                 <children>
                   <leafNode name="blind">
                     <properties>
                       <help>Disables flow isolation, all traffic passes through a single queue</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                   <leafNode name="src-host">
                     <properties>
                       <help>Flows are defined only by source address</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                   <leafNode name="dst-host">
                     <properties>
                       <help>Flows are defined only by destination address</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                   <leafNode name="host">
                     <properties>
                       <help>Flows are defined by source-destination host pairs</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                   <leafNode name="flow">
                     <properties>
                       <help>Flows are defined by the entire 5-tuple</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                   <leafNode name="dual-src-host">
                     <properties>
                       <help>Flows are defined by the 5-tuple, fairness is applied first over source addresses, then over individual flows</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                   <leafNode name="dual-dst-host">
                     <properties>
                       <help>Flows are defined by the 5-tuple, fairness is applied first over destination addresses, then over individual flows</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                   <leafNode name="triple-isolate">
                     <properties>
                       <help>Flows are defined by the 5-tuple, fairness is applied over source and destination addresses and also over individual flows (default)</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                   <leafNode name="nat">
                     <properties>
                       <help>Perform NAT lookup before applying flow-isolation rules</help>
                       <valueless/>
                     </properties>
                   </leafNode>
                 </children>
               </node>
               <leafNode name="rtt">
                 <properties>
                   <help>Round-Trip-Time for Active Queue Management (AQM)</help>
                   <valueHelp>
                     <format>u32:1-3600000</format>
                     <description>RTT in ms</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 1-3600000"/>
                   </constraint>
                   <constraintErrorMessage>RTT must be in range 1 to 3600000 milli-seconds</constraintErrorMessage>
                 </properties>
                 <defaultValue>100</defaultValue>
               </leafNode>
             </children>
           </tagNode>
           <tagNode name="drop-tail">
             <properties>
               <help>Packet limited First In, First Out queue</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               #include <include/qos/queue-limit-1-4294967295.xml.i>
             </children>
           </tagNode>
           <tagNode name="fair-queue">
             <properties>
               <help>Stochastic Fairness Queueing</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               <leafNode name="hash-interval">
                 <properties>
                   <help>Interval in seconds for queue algorithm perturbation</help>
                   <valueHelp>
                     <format>u32:0</format>
                     <description>No perturbation</description>
                   </valueHelp>
                   <valueHelp>
                     <format>u32:1-127</format>
                     <description>Interval in seconds for queue algorithm perturbation (advised: 10)</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 0-127"/>
                   </constraint>
                   <constraintErrorMessage>Interval must be in range 0 to 127</constraintErrorMessage>
                 </properties>
                 <defaultValue>0</defaultValue>
               </leafNode>
               <leafNode name="queue-limit">
                 <properties>
                   <help>Upper limit of the SFQ</help>
                   <valueHelp>
                     <format>u32:1-127</format>
                     <description>Queue size in packets</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 1-127"/>
                   </constraint>
                   <constraintErrorMessage>Queue limit must be in range 1 to 127</constraintErrorMessage>
                 </properties>
                 <defaultValue>127</defaultValue>
               </leafNode>
             </children>
           </tagNode>
           <tagNode name="fq-codel">
             <properties>
               <help>Fair Queuing (FQ) with Controlled Delay (CoDel)</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               #include <include/qos/codel-quantum.xml.i>
               #include <include/qos/flows.xml.i>
               #include <include/qos/interval.xml.i>
               #include <include/qos/queue-limit-2-10999.xml.i>
               #include <include/qos/target.xml.i>
             </children>
           </tagNode>
           <tagNode name="limiter">
             <properties>
               <help>Traffic input limiting policy</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               <tagNode name="class">
                 <properties>
                   <help>Class ID</help>
                   <valueHelp>
                     <format>u32:1-4090</format>
                     <description>Class Identifier</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 1-4090"/>
                   </constraint>
                   <constraintErrorMessage>Class identifier must be between 1 and 4090</constraintErrorMessage>
                 </properties>
                 <children>
                   #include <include/generic-description.xml.i>
                   #include <include/qos/bandwidth.xml.i>
                   #include <include/qos/burst.xml.i>
                   #include <include/qos/mtu.xml.i>
                   #include <include/qos/class-police-exceed.xml.i>
                   #include <include/qos/class-match.xml.i>
+                  #include <include/qos/class-match-group.xml.i>
                   #include <include/qos/class-priority.xml.i>
                   <leafNode name="priority">
                     <defaultValue>20</defaultValue>
                   </leafNode>
                 </children>
               </tagNode>
               <node name="default">
                 <properties>
                   <help>Default policy</help>
                 </properties>
                 <children>
                   #include <include/qos/bandwidth.xml.i>
                   #include <include/qos/burst.xml.i>
                   #include <include/qos/mtu.xml.i>
                   #include <include/qos/class-police-exceed.xml.i>
                 </children>
               </node>
             </children>
           </tagNode>
           <tagNode name="network-emulator">
             <properties>
               <help>Network emulator policy</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               #include <include/qos/bandwidth.xml.i>
               <leafNode name="delay">
                 <properties>
                   <help>Adds delay to packets outgoing to chosen network interface</help>
                   <valueHelp>
                     <format>&lt;number&gt;</format>
                     <description>Time in milliseconds</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 0-65535"/>
                   </constraint>
                   <constraintErrorMessage>Priority must be between 0 and 65535</constraintErrorMessage>
                 </properties>
               </leafNode>
               <leafNode name="corruption">
                 <properties>
                   <help>Introducing error in a random position for chosen percent of packets</help>
                   <valueHelp>
                     <format>&lt;number&gt;</format>
                     <description>Percentage of packets affected</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 0-100"/>
                   </constraint>
                   <constraintErrorMessage>Priority must be between 0 and 100</constraintErrorMessage>
                 </properties>
               </leafNode>
               <leafNode name="duplicate">
                 <properties>
                   <help>Cosen percent of packets is duplicated before queuing them</help>
                   <valueHelp>
                     <format>&lt;number&gt;</format>
                     <description>Percentage of packets affected</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 0-100"/>
                   </constraint>
                   <constraintErrorMessage>Priority must be between 0 and 100</constraintErrorMessage>
                 </properties>
               </leafNode>
               <leafNode name="loss">
                 <properties>
                   <help>Add independent loss probability to the packets outgoing to chosen network interface</help>
                   <valueHelp>
                     <format>&lt;number&gt;</format>
                     <description>Percentage of packets affected</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 0-100"/>
                   </constraint>
                   <constraintErrorMessage>Must be between 0 and 100</constraintErrorMessage>
                 </properties>
               </leafNode>
               <leafNode name="reordering">
                 <properties>
                   <help>Emulated packet reordering percentage</help>
                   <valueHelp>
                     <format>&lt;number&gt;</format>
                     <description>Percentage of packets affected</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 0-100"/>
                   </constraint>
                   <constraintErrorMessage>Must be between 0 and 100</constraintErrorMessage>
                 </properties>
               </leafNode>
               #include <include/qos/queue-limit-1-4294967295.xml.i>
             </children>
           </tagNode>
           <tagNode name="priority-queue">
             <properties>
               <help>Priority queuing based policy</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               <tagNode name="class">
                 <properties>
                   <help>Class Handle</help>
                   <valueHelp>
                     <format>u32:1-7</format>
                     <description>Priority</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 1-7"/>
                   </constraint>
                   <constraintErrorMessage>Class handle must be between 1 and 7</constraintErrorMessage>
                 </properties>
                 <children>
                   #include <include/generic-description.xml.i>
                   #include <include/qos/codel-quantum.xml.i>
                   #include <include/qos/flows.xml.i>
                   #include <include/qos/interval.xml.i>
                   #include <include/qos/class-match.xml.i>
+                  #include <include/qos/class-match-group.xml.i>
                   #include <include/qos/queue-limit-1-4294967295.xml.i>
                   #include <include/qos/queue-type.xml.i>
                   <leafNode name="queue-type">
                     <defaultValue>drop-tail</defaultValue>
                   </leafNode>
                   #include <include/qos/target.xml.i>
                 </children>
               </tagNode>
               <node name="default">
                 <properties>
                   <help>Default policy</help>
                 </properties>
                 <children>
                   #include <include/qos/codel-quantum.xml.i>
                   #include <include/qos/flows.xml.i>
                   #include <include/qos/interval.xml.i>
                   #include <include/qos/queue-limit-1-4294967295.xml.i>
                   #include <include/qos/queue-type.xml.i>
                   <leafNode name="queue-type">
                     <defaultValue>drop-tail</defaultValue>
                   </leafNode>
                   #include <include/qos/target.xml.i>
                 </children>
               </node>
             </children>
           </tagNode>
           <tagNode name="random-detect">
             <properties>
               <help>Weighted Random Early Detect policy</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               #include <include/qos/bandwidth-auto.xml.i>
               <tagNode name="precedence">
                 <properties>
                   <help>IP precedence</help>
                   <valueHelp>
                     <format>u32:0-7</format>
                     <description>IP precedence value</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 0-7"/>
                   </constraint>
                   <constraintErrorMessage>IP precedence value must be between 0 and 7</constraintErrorMessage>
                 </properties>
                 <children>
                   #include <include/qos/queue-limit-1-4294967295.xml.i>
                   #include <include/qos/queue-average-packet.xml.i>
                   #include <include/qos/queue-maximum-threshold.xml.i>
                   #include <include/qos/queue-minimum-threshold.xml.i>
                   #include <include/qos/queue-mark-probability.xml.i>
                 </children>
               </tagNode>
             </children>
           </tagNode>
           <tagNode name="rate-control">
             <properties>
               <help>Rate limiting policy (Token Bucket Filter)</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               #include <include/qos/bandwidth.xml.i>
               #include <include/qos/burst.xml.i>
               <leafNode name="latency">
                 <properties>
                   <help>Maximum latency</help>
                   <valueHelp>
                     <format>&lt;number&gt;</format>
                     <description>Time in milliseconds</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 0-4096"/>
                   </constraint>
                   <constraintErrorMessage>Threshold must be between 0 and 4096</constraintErrorMessage>
                 </properties>
                 <defaultValue>50</defaultValue>
               </leafNode>
             </children>
           </tagNode>
           <tagNode name="round-robin">
             <properties>
               <help>Deficit Round Robin Scheduler</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               <tagNode name="class">
                 <properties>
                   <help>Class ID</help>
                   <valueHelp>
                     <format>u32:1-4095</format>
                     <description>Class Identifier</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 1-4095"/>
                   </constraint>
                   <constraintErrorMessage>Class identifier must be between 1 and 4095</constraintErrorMessage>
                 </properties>
                 <children>
                   #include <include/generic-description.xml.i>
                   #include <include/qos/codel-quantum.xml.i>
                   #include <include/qos/flows.xml.i>
                   #include <include/qos/interval.xml.i>
                   #include <include/qos/class-match.xml.i>
+                  #include <include/qos/class-match-group.xml.i>
+
                   <leafNode name="quantum">
                     <properties>
                       <help>Packet scheduling quantum</help>
                       <valueHelp>
                         <format>u32:1-4294967295</format>
                         <description>Packet scheduling quantum (bytes)</description>
                       </valueHelp>
                       <constraint>
                         <validator name="numeric" argument="--range 1-4294967295"/>
                       </constraint>
                       <constraintErrorMessage>Quantum must be in range 1 to 4294967295</constraintErrorMessage>
                     </properties>
                   </leafNode>
                   #include <include/qos/queue-limit-1-4294967295.xml.i>
                   #include <include/qos/queue-type.xml.i>
                   <leafNode name="queue-type">
                     <defaultValue>drop-tail</defaultValue>
                   </leafNode>
                   #include <include/qos/target.xml.i>
                 </children>
               </tagNode>
               <node name="default">
                 <properties>
                   <help>Default policy</help>
                 </properties>
                 <children>
                   #include <include/qos/codel-quantum.xml.i>
                   #include <include/qos/flows.xml.i>
                   #include <include/qos/interval.xml.i>
                   #include <include/qos/queue-limit-1-4294967295.xml.i>
                   #include <include/qos/queue-type.xml.i>
                   <leafNode name="queue-type">
                     <defaultValue>fair-queue</defaultValue>
                   </leafNode>
                   #include <include/qos/target.xml.i>
                 </children>
               </node>
             </children>
           </tagNode>
           <tagNode name="shaper">
             <properties>
               <help>Traffic shaping based policy (Hierarchy Token Bucket)</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               #include <include/qos/bandwidth-auto.xml.i>
               <tagNode name="class">
                 <properties>
                   <help>Class ID</help>
                   <valueHelp>
                     <format>u32:2-4095</format>
                     <description>Class Identifier</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 2-4095"/>
                   </constraint>
                   <constraintErrorMessage>Class identifier must be between 2 and 4095</constraintErrorMessage>
                 </properties>
                 <children>
                   #include <include/generic-description.xml.i>
                   #include <include/qos/bandwidth-auto.xml.i>
                   #include <include/qos/burst.xml.i>
                   <leafNode name="ceiling">
                     <properties>
                       <help>Bandwidth limit for this class</help>
                       <valueHelp>
                         <format>&lt;number&gt;</format>
                         <description>Rate in kbit (kilobit per second)</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;%%</format>
                         <description>Percentage of overall rate</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;bit</format>
                         <description>bit(1), kbit(10^3), mbit(10^6), gbit, tbit</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;ibit</format>
                         <description>kibit(1024), mibit(1024^2), gibit(1024^3), tbit(1024^4)</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;ibps</format>
                         <description>kibps(1024*8), mibps(1024^2*8), gibps, tibps - Byte/sec</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;bps</format>
                         <description>bps(8),kbps(8*10^3),mbps(8*10^6), gbps, tbps - Byte/sec</description>
                       </valueHelp>
                     </properties>
                   </leafNode>
                   #include <include/qos/codel-quantum.xml.i>
                   #include <include/qos/flows.xml.i>
                   #include <include/qos/interval.xml.i>
                   #include <include/qos/class-match.xml.i>
+                  #include <include/qos/class-match-group.xml.i>
                   #include <include/qos/class-priority.xml.i>
                   #include <include/qos/queue-average-packet.xml.i>
                   #include <include/qos/queue-maximum-threshold.xml.i>
                   #include <include/qos/queue-minimum-threshold.xml.i>
                   #include <include/qos/queue-mark-probability.xml.i>
                   #include <include/qos/queue-limit-1-4294967295.xml.i>
                   #include <include/qos/queue-type.xml.i>
                   <leafNode name="queue-type">
                     <defaultValue>fq-codel</defaultValue>
                   </leafNode>
                   #include <include/qos/set-dscp.xml.i>
                   #include <include/qos/target.xml.i>
                 </children>
               </tagNode>
               <node name="default">
                 <properties>
                   <help>Default policy</help>
                 </properties>
                 <children>
                   #include <include/qos/bandwidth.xml.i>
                   #include <include/qos/burst.xml.i>
                   <leafNode name="ceiling">
                     <properties>
                       <help>Bandwidth limit for this class</help>
                       <valueHelp>
                         <format>&lt;number&gt;</format>
                         <description>Rate in kbit (kilobit per second)</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;%%</format>
                         <description>Percentage of overall rate</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;bit</format>
                         <description>bit(1), kbit(10^3), mbit(10^6), gbit, tbit</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;ibit</format>
                         <description>kibit(1024), mibit(1024^2), gibit(1024^3), tbit(1024^4)</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;ibps</format>
                         <description>kibps(1024*8), mibps(1024^2*8), gibps, tibps - Byte/sec</description>
                       </valueHelp>
                       <valueHelp>
                         <format>&lt;number&gt;bps</format>
                         <description>bps(8),kbps(8*10^3),mbps(8*10^6), gbps, tbps - Byte/sec</description>
                       </valueHelp>
                     </properties>
                   </leafNode>
                   #include <include/qos/codel-quantum.xml.i>
                   #include <include/qos/flows.xml.i>
                   #include <include/qos/interval.xml.i>
                   <leafNode name="priority">
                     <properties>
                       <help>Priority for usage of excess bandwidth</help>
                       <valueHelp>
                         <format>u32:0-7</format>
                         <description>Priority order for bandwidth pool</description>
                       </valueHelp>
                       <constraint>
                         <validator name="numeric" argument="--range 0-7"/>
                       </constraint>
                       <constraintErrorMessage>Priority must be between 0 and 7</constraintErrorMessage>
                     </properties>
                     <defaultValue>20</defaultValue>
                   </leafNode>
                   #include <include/qos/queue-average-packet.xml.i>
                   #include <include/qos/queue-maximum-threshold.xml.i>
                   #include <include/qos/queue-minimum-threshold.xml.i>
                   #include <include/qos/queue-mark-probability.xml.i>
                   #include <include/qos/queue-limit-1-4294967295.xml.i>
                   #include <include/qos/queue-type.xml.i>
                   <leafNode name="queue-type">
                     <defaultValue>fq-codel</defaultValue>
                   </leafNode>
                   #include <include/qos/set-dscp.xml.i>
                   #include <include/qos/target.xml.i>
                 </children>
               </node>
             </children>
           </tagNode>
           <tagNode name="shaper-hfsc">
             <properties>
               <help>Hierarchical Fair Service Curve's policy</help>
               <valueHelp>
                 <format>txt</format>
                 <description>Policy name</description>
               </valueHelp>
               <constraint>
                 <regex>[[:alnum:]][-_[:alnum:]]*</regex>
               </constraint>
               <constraintErrorMessage>Only alpha-numeric policy name allowed</constraintErrorMessage>
             </properties>
             <children>
               #include <include/generic-description.xml.i>
               #include <include/qos/bandwidth-auto.xml.i>
               <tagNode name="class">
                 <properties>
                   <help>Class ID</help>
                   <valueHelp>
                     <format>u32:1-4095</format>
                     <description>Class Identifier</description>
                   </valueHelp>
                   <constraint>
                     <validator name="numeric" argument="--range 1-4095"/>
                   </constraint>
                   <constraintErrorMessage>Class identifier must be between 1 and 4095</constraintErrorMessage>
                 </properties>
                 <children>
                   #include <include/generic-description.xml.i>
                   <node name="linkshare">
                     <properties>
                       <help>Linkshare class settings</help>
                     </properties>
                     <children>
                       #include <include/qos/hfsc-d.xml.i>
                       #include <include/qos/hfsc-m1.xml.i>
                       #include <include/qos/hfsc-m2.xml.i>
                     </children>
                   </node>
                   #include <include/qos/class-match.xml.i>
+                  #include <include/qos/class-match-group.xml.i>
                   <node name="realtime">
                     <properties>
                       <help>Realtime class settings</help>
                     </properties>
                     <children>
                       #include <include/qos/hfsc-d.xml.i>
                       #include <include/qos/hfsc-m1.xml.i>
                       #include <include/qos/hfsc-m2.xml.i>
                     </children>
                   </node>
                   <node name="upperlimit">
                     <properties>
                       <help>Upperlimit class settings</help>
                     </properties>
                     <children>
                       #include <include/qos/hfsc-d.xml.i>
                       #include <include/qos/hfsc-m1.xml.i>
                       #include <include/qos/hfsc-m2.xml.i>
                     </children>
                   </node>
                 </children>
               </tagNode>
               <node name="default">
                 <properties>
                   <help>Default policy</help>
                 </properties>
                 <children>
                   <node name="linkshare">
                     <properties>
                       <help>Linkshare class settings</help>
                     </properties>
                     <children>
                       #include <include/qos/hfsc-d.xml.i>
                       #include <include/qos/hfsc-m1.xml.i>
                       #include <include/qos/hfsc-m2.xml.i>
                     </children>
                   </node>
                   <node name="realtime">
                     <properties>
                       <help>Realtime class settings</help>
                     </properties>
                     <children>
                       #include <include/qos/hfsc-d.xml.i>
                       #include <include/qos/hfsc-m1.xml.i>
                       #include <include/qos/hfsc-m2.xml.i>
                     </children>
                   </node>
                   <node name="upperlimit">
                     <properties>
                       <help>Upperlimit class settings</help>
                     </properties>
                     <children>
                       #include <include/qos/hfsc-d.xml.i>
                       #include <include/qos/hfsc-m1.xml.i>
                       #include <include/qos/hfsc-m2.xml.i>
                     </children>
                   </node>
                 </children>
               </node>
             </children>
           </tagNode>
         </children>
       </node>
+      <tagNode name="traffic-match-group">
+        <properties>
+          <help>Filter group for QoS policy</help>
+          <valueHelp>
+            <format>txt</format>
+            <description>Match group name</description>
+          </valueHelp>
+          <constraint>
+            <regex>[^-].*</regex>
+          </constraint>
+          <constraintErrorMessage>Match group name cannot start with hyphen</constraintErrorMessage>
+        </properties>
+          <children>
+            #include <include/generic-description.xml.i>
+            <tagNode name="match">
+              <properties>
+                <help>Class matching rule name</help>
+                <constraint>
+                  <regex>[^-].*</regex>
+                </constraint>
+                <constraintErrorMessage>Match queue name cannot start with hyphen</constraintErrorMessage>
+              </properties>
+              <children>
+                #include <include/generic-description.xml.i>
+                #include <include/qos/class-match-ipv4.xml.i>
+                #include <include/qos/class-match-ipv6.xml.i>
+                #include <include/qos/class-match-mark.xml.i>
+                #include <include/qos/class-match-vif.xml.i>
+              </children>
+            </tagNode>
+            #include <include/qos/class-match-group.xml.i>
+          </children>
+      </tagNode>
     </children>
   </node>
 </interfaceDefinition>
diff --git a/smoketest/scripts/cli/test_qos.py b/smoketest/scripts/cli/test_qos.py
index 5977b2f41..b98c0e9b7 100755
--- a/smoketest/scripts/cli/test_qos.py
+++ b/smoketest/scripts/cli/test_qos.py
@@ -1,764 +1,859 @@
 #!/usr/bin/env python3
 #
 # Copyright (C) 2022-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
 import unittest
 
 from json import loads
 from base_vyostest_shim import VyOSUnitTestSHIM
 
 from vyos.configsession import ConfigSessionError
 from vyos.ifconfig import Section
 from vyos.utils.process import cmd
 
 base_path = ['qos']
 
 def get_tc_qdisc_json(interface) -> dict:
     tmp = cmd(f'tc -detail -json qdisc show dev {interface}')
     tmp = loads(tmp)
     return next(iter(tmp))
 
 def get_tc_filter_json(interface, direction) -> list:
     if direction not in ['ingress', 'egress']:
         raise ValueError()
     tmp = cmd(f'tc -detail -json filter show dev {interface} {direction}')
     tmp = loads(tmp)
     return tmp
 
 def get_tc_filter_details(interface, direction) -> list:
     # json doesn't contain all params, such as mtu
     if direction not in ['ingress', 'egress']:
         raise ValueError()
     tmp = cmd(f'tc -details filter show dev {interface} {direction}')
     return tmp
 
 class TestQoS(VyOSUnitTestSHIM.TestCase):
     @classmethod
     def setUpClass(cls):
         super(TestQoS, cls).setUpClass()
 
         # ensure we can also run this test on a live system - so lets clean
         # out the current configuration :)
         cls.cli_delete(cls, base_path)
 
         # We only test on physical interfaces and not VLAN (sub-)interfaces
         cls._interfaces = []
         if 'TEST_ETH' in os.environ:
             tmp = os.environ['TEST_ETH'].split()
             cls._interfaces = tmp
         else:
             for tmp in Section.interfaces('ethernet', vlan=False):
                 cls._interfaces.append(tmp)
 
     def tearDown(self):
         # delete testing SSH config
         self.cli_delete(base_path)
         self.cli_commit()
 
     def test_01_cake(self):
         bandwidth = 1000000
         rtt = 200
 
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
             self.cli_set(base_path + ['policy', 'cake', policy_name, 'bandwidth', str(bandwidth)])
             self.cli_set(base_path + ['policy', 'cake', policy_name, 'rtt', str(rtt)])
             self.cli_set(base_path + ['policy', 'cake', policy_name, 'flow-isolation', 'dual-src-host'])
 
             bandwidth += 1000000
             rtt += 20
 
         # commit changes
         self.cli_commit()
 
         bandwidth = 1000000
         rtt = 200
         for interface in self._interfaces:
             tmp = get_tc_qdisc_json(interface)
 
             self.assertEqual('cake', tmp['kind'])
             # TC store rates as a 32-bit unsigned integer in bps (Bytes per second)
             self.assertEqual(int(bandwidth *125), tmp['options']['bandwidth'])
             # RTT internally is in us
             self.assertEqual(int(rtt *1000), tmp['options']['rtt'])
             self.assertEqual('dual-srchost', tmp['options']['flowmode'])
             self.assertFalse(tmp['options']['ingress'])
             self.assertFalse(tmp['options']['nat'])
             self.assertTrue(tmp['options']['raw'])
 
             bandwidth += 1000000
             rtt += 20
 
     def test_02_drop_tail(self):
         queue_limit = 50
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'ingress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
             self.cli_set(base_path + ['policy', 'drop-tail', policy_name, 'queue-limit', str(queue_limit)])
 
             queue_limit += 10
 
         # commit changes
         self.cli_commit()
 
         queue_limit = 50
         for interface in self._interfaces:
             tmp = get_tc_qdisc_json(interface)
 
             self.assertEqual('pfifo', tmp['kind'])
             self.assertEqual(queue_limit, tmp['options']['limit'])
 
             queue_limit += 10
 
     def test_03_fair_queue(self):
         hash_interval = 10
         queue_limit = 5
         policy_type = 'fair-queue'
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'ingress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'hash-interval', str(hash_interval)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'queue-limit', str(queue_limit)])
 
             hash_interval += 1
             queue_limit += 1
 
         # commit changes
         self.cli_commit()
 
         hash_interval = 10
         queue_limit = 5
         for interface in self._interfaces:
             tmp = get_tc_qdisc_json(interface)
 
             self.assertEqual('sfq', tmp['kind'])
             self.assertEqual(hash_interval, tmp['options']['perturb'])
             self.assertEqual(queue_limit, tmp['options']['limit'])
 
             hash_interval += 1
             queue_limit += 1
 
     def test_04_fq_codel(self):
         policy_type = 'fq-codel'
         codel_quantum = 1500
         flows = 512
         interval = 100
         queue_limit = 2048
         target = 5
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'ingress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'codel-quantum', str(codel_quantum)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'flows', str(flows)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'interval', str(interval)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'queue-limit', str(queue_limit)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'target', str(target)])
 
             codel_quantum += 10
             flows += 2
             interval += 10
             queue_limit += 512
             target += 1
 
         # commit changes
         self.cli_commit()
 
         codel_quantum = 1500
         flows = 512
         interval = 100
         queue_limit = 2048
         target = 5
         for interface in self._interfaces:
             tmp = get_tc_qdisc_json(interface)
 
             self.assertEqual('fq_codel', tmp['kind'])
             self.assertEqual(codel_quantum, tmp['options']['quantum'])
             self.assertEqual(flows, tmp['options']['flows'])
             self.assertEqual(queue_limit, tmp['options']['limit'])
 
             # due to internal rounding we need to substract 1 from interval and target after converting to milliseconds
             # configuration of:
             # tc qdisc add dev eth0 root fq_codel quantum 1500 flows 512 interval 100ms limit 2048 target 5ms noecn
             # results in: tc -j qdisc show dev eth0
             # [{"kind":"fq_codel","handle":"8046:","root":true,"refcnt":3,"options":{"limit":2048,"flows":512,
             #   "quantum":1500,"target":4999,"interval":99999,"memory_limit":33554432,"drop_batch":64}}]
             self.assertAlmostEqual(tmp['options']['interval'], interval *1000, delta=1)
             self.assertAlmostEqual(tmp['options']['target'], target *1000 -1, delta=1)
 
             codel_quantum += 10
             flows += 2
             interval += 10
             queue_limit += 512
             target += 1
 
     def test_05_limiter(self):
         qos_config = {
             '1' : {
                 'bandwidth' : '3000000',
                 'exceed' : 'pipe',
                 'burst' : '100Kb',
                 'mtu' : '1600',
                 'not-exceed' : 'continue',
                 'priority': '15',
                 'match4' : {
                     'ssh'   : { 'dport' : '22', },
                     },
                 },
             '2' : {
                 'bandwidth' : '1000000',
                 'match6' : {
                     'ssh'   : { 'dport' : '22', },
                     },
                 },
             }
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'egress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
             # set default bandwidth parameter for all remaining connections
             self.cli_set(base_path + ['policy', 'limiter', policy_name, 'default', 'bandwidth', '500000'])
             self.cli_set(base_path + ['policy', 'limiter', policy_name, 'default', 'burst', '200kb'])
             self.cli_set(base_path + ['policy', 'limiter', policy_name, 'default', 'exceed', 'drop'])
             self.cli_set(base_path + ['policy', 'limiter', policy_name, 'default', 'mtu', '3000'])
             self.cli_set(base_path + ['policy', 'limiter', policy_name, 'default', 'not-exceed', 'ok'])
 
             for qos_class, qos_class_config in qos_config.items():
                 qos_class_base = base_path + ['policy', 'limiter', policy_name, 'class', qos_class]
 
                 if 'match4' in qos_class_config:
                     for match, match_config in qos_class_config['match4'].items():
                         if 'dport' in match_config:
                             self.cli_set(qos_class_base + ['match', match, 'ip', 'destination', 'port', match_config['dport']])
 
                 if 'match6' in qos_class_config:
                     for match, match_config in qos_class_config['match6'].items():
                         if 'dport' in match_config:
                             self.cli_set(qos_class_base + ['match', match, 'ipv6', 'destination', 'port', match_config['dport']])
 
                 if 'bandwidth' in qos_class_config:
                     self.cli_set(qos_class_base + ['bandwidth', qos_class_config['bandwidth']])
 
                 if 'exceed' in qos_class_config:
                     self.cli_set(qos_class_base + ['exceed', qos_class_config['exceed']])
 
                 if 'not-exceed' in qos_class_config:
                     self.cli_set(qos_class_base + ['not-exceed', qos_class_config['not-exceed']])
 
                 if 'burst' in qos_class_config:
                     self.cli_set(qos_class_base + ['burst', qos_class_config['burst']])
 
                 if 'mtu' in qos_class_config:
                     self.cli_set(qos_class_base + ['mtu', qos_class_config['mtu']])
 
                 if 'priority' in qos_class_config:
                     self.cli_set(qos_class_base + ['priority', qos_class_config['priority']])
 
 
         # commit changes
         self.cli_commit()
 
         for interface in self._interfaces:
             for filter in get_tc_filter_json(interface, 'ingress'):
                 # bail out early if filter has no attached action
                 if 'options' not in filter or 'actions' not in filter['options']:
                     continue
 
                 for qos_class, qos_class_config in qos_config.items():
                     # Every flowid starts with ffff and we encopde the class number after the colon
                     if 'flowid' not in filter['options'] or filter['options']['flowid'] != f'ffff:{qos_class}':
                         continue
 
                     ip_hdr_offset = 20
                     if 'match6' in qos_class_config:
                         ip_hdr_offset = 40
 
                     self.assertEqual(ip_hdr_offset, filter['options']['match']['off'])
                     if 'dport' in match_config:
                         dport = int(match_config['dport'])
                         self.assertEqual(f'{dport:x}', filter['options']['match']['value'])
 
             tc_details = get_tc_filter_details(interface, 'ingress')
             self.assertTrue('filter parent ffff: protocol all pref 20 u32 chain 0' in tc_details)
             self.assertTrue('rate 1Gbit burst 15125b mtu 2Kb action drop overhead 0b linklayer ethernet' in tc_details)
             self.assertTrue('filter parent ffff: protocol all pref 15 u32 chain 0' in tc_details)
             self.assertTrue('rate 3Gbit burst 102000b mtu 1600b action pipe/continue overhead 0b linklayer ethernet' in tc_details)
             self.assertTrue('rate 500Mbit burst 204687b mtu 3000b action drop overhead 0b linklayer ethernet' in tc_details)
             self.assertTrue('filter parent ffff: protocol all pref 255 basic chain 0' in tc_details)
 
     def test_06_network_emulator(self):
         policy_type = 'network-emulator'
 
         bandwidth = 1000000
         corruption = 1
         delay = 2
         duplicate = 3
         loss = 4
         queue_limit = 5
         reordering = 6
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'ingress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
 
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'bandwidth', str(bandwidth)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'corruption', str(corruption)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'delay', str(delay)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'duplicate', str(duplicate)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'loss', str(loss)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'queue-limit', str(queue_limit)])
             self.cli_set(base_path + ['policy', policy_type, policy_name, 'reordering', str(reordering)])
 
             bandwidth += 1000000
             corruption += 1
             delay += 1
             duplicate +=1
             loss += 1
             queue_limit += 1
             reordering += 1
 
         # commit changes
         self.cli_commit()
 
         bandwidth = 1000000
         corruption = 1
         delay = 2
         duplicate = 3
         loss = 4
         queue_limit = 5
         reordering = 6
         for interface in self._interfaces:
             tmp = get_tc_qdisc_json(interface)
             self.assertEqual('netem', tmp['kind'])
 
             self.assertEqual(int(bandwidth *125), tmp['options']['rate']['rate'])
             # values are in %
             self.assertEqual(corruption/100, tmp['options']['corrupt']['corrupt'])
             self.assertEqual(duplicate/100, tmp['options']['duplicate']['duplicate'])
             self.assertEqual(loss/100, tmp['options']['loss-random']['loss'])
             self.assertEqual(reordering/100, tmp['options']['reorder']['reorder'])
             self.assertEqual(delay/1000, tmp['options']['delay']['delay'])
 
             self.assertEqual(queue_limit, tmp['options']['limit'])
 
             bandwidth += 1000000
             corruption += 1
             delay += 1
             duplicate += 1
             loss += 1
             queue_limit += 1
             reordering += 1
 
     def test_07_priority_queue(self):
         priorities = ['1', '2', '3', '4', '5']
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'ingress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
             self.cli_set(base_path + ['policy', 'priority-queue', policy_name, 'default', 'queue-limit', '10'])
 
             for priority in priorities:
                 prio_base = base_path + ['policy', 'priority-queue', policy_name, 'class', priority]
                 self.cli_set(prio_base + ['match', f'prio-{priority}', 'ip', 'destination', 'port', str(1000 + int(priority))])
 
         # commit changes
         self.cli_commit()
 
     def test_08_random_detect(self):
         bandwidth = 5000
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'ingress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
             self.cli_set(base_path + ['policy', 'random-detect', policy_name, 'bandwidth', str(bandwidth)])
 
             bandwidth += 1000
 
         # commit changes
         self.cli_commit()
 
         bandwidth = 5000
         for interface in self._interfaces:
             tmp = get_tc_qdisc_json(interface)
             self.assertTrue('gred' in tmp.get('kind'))
             self.assertEqual(8, len(tmp.get('options', {}).get('vqs')))
             self.assertEqual(8, tmp.get('options', {}).get('dp_cnt'))
             self.assertEqual(0, tmp.get('options', {}).get('dp_default'))
             self.assertTrue(tmp.get('options', {}).get('grio'))
 
     def test_09_rate_control(self):
         bandwidth = 5000
         burst = 20
         latency = 5
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'ingress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
             self.cli_set(base_path + ['policy', 'rate-control', policy_name, 'bandwidth', str(bandwidth)])
             self.cli_set(base_path + ['policy', 'rate-control', policy_name, 'burst', str(burst)])
             self.cli_set(base_path + ['policy', 'rate-control', policy_name, 'latency', str(latency)])
 
             bandwidth += 1000
             burst += 5
             latency += 1
         # commit changes
         self.cli_commit()
 
         bandwidth = 5000
         burst = 20
         latency = 5
         for interface in self._interfaces:
             tmp = get_tc_qdisc_json(interface)
 
             self.assertEqual('tbf', tmp['kind'])
             self.assertEqual(0, tmp['options']['mpu'])
             # TC store rates as a 32-bit unsigned integer in bps (Bytes per second)
             self.assertEqual(int(bandwidth * 125), tmp['options']['rate'])
 
             bandwidth += 1000
             burst += 5
             latency += 1
 
     def test_10_round_robin(self):
         qos_config = {
             '1' : {
                 'match4' : {
                     'ssh'   : { 'dport' : '22', },
                     },
                 },
             '2' : {
                 'match6' : {
                     'ssh'   : { 'dport' : '22', },
                     },
                 },
             }
 
         first = True
         for interface in self._interfaces:
             policy_name = f'qos-policy-{interface}'
 
             if first:
                 self.cli_set(base_path + ['interface', interface, 'ingress', policy_name])
                 # verify() - selected QoS policy on interface only supports egress
                 with self.assertRaises(ConfigSessionError):
                     self.cli_commit()
                 self.cli_delete(base_path + ['interface', interface, 'ingress', policy_name])
                 first = False
 
             self.cli_set(base_path + ['interface', interface, 'egress', policy_name])
 
             for qos_class, qos_class_config in qos_config.items():
                 qos_class_base = base_path + ['policy', 'round-robin', policy_name, 'class', qos_class]
 
                 if 'match4' in qos_class_config:
                     for match, match_config in qos_class_config['match4'].items():
                         if 'dport' in match_config:
                             self.cli_set(qos_class_base + ['match', match, 'ip', 'destination', 'port', match_config['dport']])
 
                 if 'match6' in qos_class_config:
                     for match, match_config in qos_class_config['match6'].items():
                         if 'dport' in match_config:
                             self.cli_set(qos_class_base + ['match', match, 'ipv6', 'destination', 'port', match_config['dport']])
 
 
         # commit changes
         self.cli_commit()
 
         for interface in self._interfaces:
             tmp = get_tc_qdisc_json(interface)
             self.assertEqual('drr', tmp['kind'])
 
             for filter in get_tc_filter_json(interface, 'ingress'):
                 # bail out early if filter has no attached action
                 if 'options' not in filter or 'actions' not in filter['options']:
                     continue
 
                 for qos_class, qos_class_config in qos_config.items():
                     # Every flowid starts with ffff and we encopde the class number after the colon
                     if 'flowid' not in filter['options'] or filter['options']['flowid'] != f'ffff:{qos_class}':
                         continue
 
                     ip_hdr_offset = 20
                     if 'match6' in qos_class_config:
                         ip_hdr_offset = 40
 
                     self.assertEqual(ip_hdr_offset, filter['options']['match']['off'])
                     if 'dport' in match_config:
                         dport = int(match_config['dport'])
                         self.assertEqual(f'{dport:x}', filter['options']['match']['value'])
 
     def test_11_shaper(self):
         bandwidth = 250
         default_bandwidth = 20
         default_ceil = 30
         class_bandwidth = 50
         class_ceil = 80
         dst_address = '192.0.2.8/32'
 
         for interface in self._interfaces:
             shaper_name = f'qos-shaper-{interface}'
 
             self.cli_set(base_path + ['interface', interface, 'egress', shaper_name])
             self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'bandwidth', f'{bandwidth}mbit'])
             self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'bandwidth', f'{default_bandwidth}mbit'])
             self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'ceiling', f'{default_ceil}mbit'])
             self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'queue-type', 'fair-queue'])
             self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '23', 'bandwidth', f'{class_bandwidth}mbit'])
             self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '23', 'ceiling', f'{class_ceil}mbit'])
             self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '23', 'match', '10', 'ip', 'destination', 'address', dst_address])
 
             bandwidth += 1
             default_bandwidth += 1
             default_ceil += 1
             class_bandwidth += 1
             class_ceil += 1
 
         # commit changes
         self.cli_commit()
 
         bandwidth = 250
         default_bandwidth = 20
         default_ceil = 30
         class_bandwidth = 50
         class_ceil = 80
 
         for interface in self._interfaces:
             config_entries = (
                 f'root rate {bandwidth}Mbit ceil {bandwidth}Mbit',
                 f'prio 0 rate {class_bandwidth}Mbit ceil {class_ceil}Mbit',
                 f'prio 7 rate {default_bandwidth}Mbit ceil {default_ceil}Mbit'
             )
 
             output = cmd(f'tc class show dev {interface}')
 
             for config_entry in config_entries:
                 self.assertIn(config_entry, output)
 
             bandwidth += 1
             default_bandwidth += 1
             default_ceil += 1
             class_bandwidth += 1
             class_ceil += 1
 
     def test_12_shaper_with_red_queue(self):
         bandwidth = 100
         default_bandwidth = 100
         default_burst = 100
         interface = self._interfaces[0]
         class_bandwidth = 50
         dst_address = '192.0.2.8/32'
 
         shaper_name = f'qos-shaper-{interface}'
         self.cli_set(base_path + ['interface', interface, 'egress', shaper_name])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'bandwidth', f'{bandwidth}mbit'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'bandwidth', f'{default_bandwidth}%'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'burst', f'{default_burst}'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'queue-type', 'random-detect'])
 
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '2', 'bandwidth', f'{class_bandwidth}mbit'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '2', 'match', '10', 'ip', 'destination', 'address', dst_address])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '2', 'queue-type', 'random-detect'])
 
         # commit changes
         self.cli_commit()
 
         # check root htb config
         output = cmd(f'tc class show dev {interface}')
 
         config_entries = (
             f'prio 0 rate {class_bandwidth}Mbit ceil 50Mbit burst 15Kb',  # specified class
             f'prio 7 rate {default_bandwidth}Mbit ceil 100Mbit burst {default_burst}b',  # default class
         )
         for config_entry in config_entries:
             self.assertIn(config_entry, output)
 
         output = cmd(f'tc -d qdisc show dev {interface}')
         config_entries = (
             'qdisc red',  # use random detect
             'limit 72Kb min 9Kb max 18Kb ewma 3 probability 0.1',  # default config for random detect
         )
         for config_entry in config_entries:
             self.assertIn(config_entry, output)
 
         # test random detect queue params
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'queue-limit', '1024'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'average-packet', '1024'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'maximum-threshold', '32'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'minimum-threshold', '16'])
 
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '2', 'queue-limit', '1024'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '2', 'average-packet', '512'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '2', 'maximum-threshold', '32'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '2', 'minimum-threshold', '16'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '2', 'mark-probability', '20'])
 
         self.cli_commit()
 
         output = cmd(f'tc -d qdisc show dev {interface}')
         config_entries = (
             'qdisc red',  # use random detect
             'limit 1Mb min 16Kb max 32Kb ewma 3 probability 0.1',  # default config for random detect
             'limit 512Kb min 8Kb max 16Kb ewma 3 probability 0.05',  # class config for random detect
         )
         for config_entry in config_entries:
             self.assertIn(config_entry, output)
 
     def test_13_shaper_delete_only_rule(self):
         default_bandwidth = 100
         default_burst = 100
         interface = self._interfaces[0]
         class_bandwidth = 50
         class_ceiling = 5
         src_address = '10.1.1.0/24'
 
         shaper_name = f'qos-shaper-{interface}'
         self.cli_set(base_path + ['interface', interface, 'egress', shaper_name])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'bandwidth', f'10mbit'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'bandwidth', f'{default_bandwidth}mbit'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'default', 'burst', f'{default_burst}'])
 
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '30', 'bandwidth', f'{class_bandwidth}mbit'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '30', 'ceiling', f'{class_ceiling}mbit'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '30', 'match', 'ADDRESS30', 'ip', 'source', 'address', src_address])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '30', 'match', 'ADDRESS30', 'description', 'smoketest'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '30', 'priority', '5'])
         self.cli_set(base_path + ['policy', 'shaper', shaper_name, 'class', '30', 'queue-type', 'fair-queue'])
 
         # commit changes
         self.cli_commit()
         # check root htb config
         output = cmd(f'tc class show dev {interface}')
 
         config_entries = (
             f'prio 5 rate {class_bandwidth}Mbit ceil {class_ceiling}Mbit burst 15Kb',  # specified class
             f'prio 7 rate {default_bandwidth}Mbit ceil 100Mbit burst {default_burst}b',  # default class
         )
         for config_entry in config_entries:
             self.assertIn(config_entry, output)
 
         self.assertTrue('' != cmd(f'tc filter show dev {interface}'))
         # self.cli_delete(base_path + ['policy', 'shaper', shaper_name, 'class', '30', 'match', 'ADDRESS30'])
         self.cli_delete(base_path + ['policy', 'shaper', shaper_name, 'class', '30', 'match', 'ADDRESS30', 'ip', 'source', 'address', src_address])
         self.cli_commit()
         self.assertEqual('', cmd(f'tc filter show dev {interface}'))
 
     def test_14_policy_limiter_marked_traffic(self):
         policy_name = 'smoke_test'
         base_policy_path = ['qos', 'policy', 'limiter', policy_name]
 
         self.cli_set(['qos', 'interface', self._interfaces[0], 'ingress', policy_name])
         self.cli_set(base_policy_path + ['class', '100', 'bandwidth', '20gbit'])
         self.cli_set(base_policy_path + ['class', '100', 'burst', '3760k'])
         self.cli_set(base_policy_path + ['class', '100', 'match', 'INTERNAL', 'mark', '100'])
         self.cli_set(base_policy_path + ['class', '100', 'priority', '20'])
         self.cli_set(base_policy_path + ['default', 'bandwidth', '1gbit'])
         self.cli_set(base_policy_path + ['default', 'burst', '125000000b'])
         self.cli_commit()
 
         tc_filters = cmd(f'tc filter show dev {self._interfaces[0]} ingress')
         # class 100
         self.assertIn('filter parent ffff: protocol all pref 20 fw chain 0', tc_filters)
         self.assertIn('action order 1:  police 0x1 rate 20Gbit burst 3847500b mtu 2Kb action drop overhead 0b', tc_filters)
         # default
         self.assertIn('filter parent ffff: protocol all pref 255 basic chain 0', tc_filters)
         self.assertIn('action order 1:  police 0x2 rate 1Gbit burst 125000000b mtu 2Kb action drop overhead 0b', tc_filters)
 
+    def test_15_traffic_match_group(self):
+        interface = self._interfaces[0]
+        self.cli_set(['qos', 'interface', interface, 'egress', 'VyOS-HTB'])
+        base_policy_path = ['qos', 'policy', 'shaper', 'VyOS-HTB']
+
+        #old syntax
+        self.cli_set(base_policy_path + ['bandwidth', '100mbit'])
+        self.cli_set(base_policy_path + ['class', '10', 'bandwidth', '40%'])
+        self.cli_set(base_policy_path + ['class', '10', 'match', 'AF11', 'ip', 'dscp', 'AF11'])
+        self.cli_set(base_policy_path + ['class', '10', 'match', 'AF41', 'ip', 'dscp', 'AF41'])
+        self.cli_set(base_policy_path + ['class', '10', 'match', 'AF43', 'ip', 'dscp', 'AF43'])
+        self.cli_set(base_policy_path + ['class', '10', 'match', 'CS4', 'ip', 'dscp', 'CS4'])
+        self.cli_set(base_policy_path + ['class', '10', 'priority', '1'])
+        self.cli_set(base_policy_path + ['class', '10', 'queue-type', 'fair-queue'])
+        self.cli_set(base_policy_path + ['class', '20', 'bandwidth', '30%'])
+        self.cli_set(base_policy_path + ['class', '20', 'match', 'EF', 'ip', 'dscp', 'EF'])
+        self.cli_set(base_policy_path + ['class', '20', 'match', 'CS5', 'ip', 'dscp', 'CS5'])
+        self.cli_set(base_policy_path + ['class', '20', 'priority', '2'])
+        self.cli_set(base_policy_path + ['class', '20', 'queue-type', 'fair-queue'])
+        self.cli_set(base_policy_path + ['default', 'bandwidth', '20%'])
+        self.cli_set(base_policy_path + ['default', 'queue-type', 'fair-queue'])
+        self.cli_commit()
+
+        tc_filters_old = cmd(f'tc -details filter show dev {interface}')
+        self.assertIn('match 00280000/00ff0000', tc_filters_old)
+        self.assertIn('match 00880000/00ff0000', tc_filters_old)
+        self.assertIn('match 00980000/00ff0000', tc_filters_old)
+        self.assertIn('match 00800000/00ff0000', tc_filters_old)
+        self.assertIn('match 00a00000/00ff0000', tc_filters_old)
+        self.assertIn('match 00b80000/00ff0000', tc_filters_old)
+        # delete config by old syntax
+        self.cli_delete(base_policy_path)
+        self.cli_delete(['qos', 'interface', interface, 'egress', 'VyOS-HTB'])
+        self.cli_commit()
+        self.assertEqual('', cmd(f'tc -s filter show dev {interface}'))
+
+        self.cli_set(['qos', 'interface', interface, 'egress', 'VyOS-HTB'])
+        # prepare traffic match group
+        self.cli_set(['qos', 'traffic-match-group', 'VOICE', 'description', 'voice shaper'])
+        self.cli_set(['qos', 'traffic-match-group', 'VOICE', 'match', 'EF', 'ip', 'dscp', 'EF'])
+        self.cli_set(['qos', 'traffic-match-group', 'VOICE', 'match', 'CS5', 'ip', 'dscp', 'CS5'])
+
+        self.cli_set(['qos', 'traffic-match-group', 'REAL_TIME_COMMON', 'description', 'real time common filters'])
+        self.cli_set(['qos', 'traffic-match-group', 'REAL_TIME_COMMON', 'match', 'AF43', 'ip', 'dscp', 'AF43'])
+        self.cli_set(['qos', 'traffic-match-group', 'REAL_TIME_COMMON', 'match', 'CS4', 'ip', 'dscp', 'CS4'])
+
+        self.cli_set(['qos', 'traffic-match-group', 'REAL_TIME', 'description', 'real time shaper'])
+        self.cli_set(['qos', 'traffic-match-group', 'REAL_TIME', 'match', 'AF41', 'ip', 'dscp', 'AF41'])
+        self.cli_set(['qos', 'traffic-match-group', 'REAL_TIME', 'match-group', 'REAL_TIME_COMMON'])
+
+        # new syntax
+        self.cli_set(base_policy_path + ['bandwidth', '100mbit'])
+        self.cli_set(base_policy_path + ['class', '10', 'bandwidth', '40%'])
+        self.cli_set(base_policy_path + ['class', '10', 'match', 'AF11', 'ip', 'dscp', 'AF11'])
+        self.cli_set(base_policy_path + ['class', '10', 'match-group', 'REAL_TIME'])
+        self.cli_set(base_policy_path + ['class', '10', 'priority', '1'])
+        self.cli_set(base_policy_path + ['class', '10', 'queue-type', 'fair-queue'])
+        self.cli_set(base_policy_path + ['class', '20', 'bandwidth', '30%'])
+        self.cli_set(base_policy_path + ['class', '20', 'match-group', 'VOICE'])
+        self.cli_set(base_policy_path + ['class', '20', 'priority', '2'])
+        self.cli_set(base_policy_path + ['class', '20', 'queue-type', 'fair-queue'])
+        self.cli_set(base_policy_path + ['default', 'bandwidth', '20%'])
+        self.cli_set(base_policy_path + ['default', 'queue-type', 'fair-queue'])
+        self.cli_commit()
+
+        self.assertEqual(tc_filters_old, cmd(f'tc -details filter show dev {interface}'))
+
+    def test_16_wrong_traffic_match_group(self):
+        interface = self._interfaces[0]
+        self.cli_set(['qos', 'interface', interface])
+
+        # Can not use both IPv6 and IPv4 in one match
+        self.cli_set(['qos', 'traffic-match-group', '1', 'match', 'one', 'ip', 'dscp', 'EF'])
+        self.cli_set(['qos', 'traffic-match-group', '1', 'match', 'one', 'ipv6', 'dscp', 'EF'])
+        with self.assertRaises(ConfigSessionError) as e:
+            self.cli_commit()
+
+        # check contain itself, should commit success
+        self.cli_delete(['qos', 'traffic-match-group', '1', 'match', 'one', 'ipv6'])
+        self.cli_set(['qos', 'traffic-match-group', '1', 'match-group', '1'])
+        self.cli_commit()
+
+        # check cycle dependency, should commit success
+        self.cli_set(['qos', 'traffic-match-group', '1', 'match-group', '3'])
+        self.cli_set(['qos', 'traffic-match-group', '2', 'match', 'one', 'ip', 'dscp', 'CS4'])
+        self.cli_set(['qos', 'traffic-match-group', '2', 'match-group', '1'])
+
+        self.cli_set(['qos', 'traffic-match-group', '3', 'match', 'one', 'ipv6', 'dscp', 'CS4'])
+        self.cli_set(['qos', 'traffic-match-group', '3', 'match-group', '2'])
+        self.cli_commit()
+
+        # inherit from non exist group, should commit success with warning
+        self.cli_set(['qos', 'traffic-match-group', '3', 'match-group', 'unexpected'])
+        self.cli_commit()
+
 
 if __name__ == '__main__':
     unittest.main(verbosity=2)
diff --git a/src/completion/qos/list_traffic_match_group.py b/src/completion/qos/list_traffic_match_group.py
new file mode 100644
index 000000000..015d7ada9
--- /dev/null
+++ b/src/completion/qos/list_traffic_match_group.py
@@ -0,0 +1,35 @@
+#!/usr/bin/env python3
+#
+# Copyright (C) 2024 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/>.
+
+from vyos.config import Config
+
+
+def get_qos_traffic_match_group():
+    config = Config()
+    base = ['qos', 'traffic-match-group']
+    conf = config.get_config_dict(base, key_mangling=('-', '_'))
+    groups = []
+
+    for group in conf.get('traffic_match_group', []):
+        groups.append(group)
+
+    return groups
+
+
+if __name__ == "__main__":
+    groups = get_qos_traffic_match_group()
+    print(" ".join(groups))
+
diff --git a/src/conf_mode/qos.py b/src/conf_mode/qos.py
index 8a590cbc6..45248fb4a 100755
--- a/src/conf_mode/qos.py
+++ b/src/conf_mode/qos.py
@@ -1,265 +1,332 @@
 #!/usr/bin/env python3
 #
 # Copyright (C) 2023-2024 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/>.
 
 from sys import exit
 from netifaces import interfaces
 
+from vyos.base import Warning
 from vyos.config import Config
 from vyos.configdep import set_dependents
 from vyos.configdep import call_dependents
 from vyos.configdict import dict_merge
 from vyos.configverify import verify_interface_exists
 from vyos.ifconfig import Section
 from vyos.qos import CAKE
 from vyos.qos import DropTail
 from vyos.qos import FairQueue
 from vyos.qos import FQCodel
 from vyos.qos import Limiter
 from vyos.qos import NetEm
 from vyos.qos import Priority
 from vyos.qos import RandomDetect
 from vyos.qos import RateLimiter
 from vyos.qos import RoundRobin
 from vyos.qos import TrafficShaper
 from vyos.qos import TrafficShaperHFSC
 from vyos.utils.dict import dict_search_recursive
 from vyos.utils.process import run
 from vyos import ConfigError
 from vyos import airbag
 from vyos.xml_ref import relative_defaults
 
 
 airbag.enable()
 
 map_vyops_tc = {
     'cake'             : CAKE,
     'drop_tail'        : DropTail,
     'fair_queue'       : FairQueue,
     'fq_codel'         : FQCodel,
     'limiter'          : Limiter,
     'network_emulator' : NetEm,
     'priority_queue'   : Priority,
     'random_detect'    : RandomDetect,
     'rate_control'     : RateLimiter,
     'round_robin'      : RoundRobin,
     'shaper'           : TrafficShaper,
     'shaper_hfsc'      : TrafficShaperHFSC,
 }
 
 def get_shaper(qos, interface_config, direction):
     policy_name = interface_config[direction]
     # An interface might have a QoS configuration, search the used
     # configuration referenced by this. Path will hold the dict element
     # referenced by the config, as this will be of sort:
     #
     # ['policy', 'drop_tail', 'foo-dtail'] <- we are only interested in
     # drop_tail as the policy/shaper type
     _, path = next(dict_search_recursive(qos, policy_name))
     shaper_type = path[1]
     shaper_config = qos['policy'][shaper_type][policy_name]
 
     return (map_vyops_tc[shaper_type], shaper_config)
 
 
 def _clean_conf_dict(conf):
     """
     Delete empty nodes from config e.g.
         match ADDRESS30 {
             ip {
                 source {}
             }
         }
     """
     if isinstance(conf, dict):
         return {node: _clean_conf_dict(val) for node, val in conf.items() if val != {} and _clean_conf_dict(val) != {}}
     else:
         return conf
 
 
+def _get_group_filters(config: dict, group_name: str, visited=None) -> dict:
+    filters = dict()
+    if not visited:
+        visited = [group_name, ]
+    else:
+        if group_name in visited:
+            return filters
+        visited.append(group_name)
+
+    for filter, filter_config in config.get(group_name, {}).items():
+        if filter == 'match':
+            for match, match_config in filter_config.items():
+               filters[f'{group_name}-{match}'] = match_config
+        elif filter == 'match_group':
+            for group in filter_config:
+                filters.update(_get_group_filters(config, group, visited))
+
+    return filters
+
+
+def _get_group_match(config:dict, group_name:str) -> dict:
+    match = dict()
+    for key, val in _get_group_filters(config, group_name).items():
+        # delete duplicate matches
+        if val not in match.values():
+            match[key] = val
+
+    return match
+
+
 def get_config(config=None):
     if config:
         conf = config
     else:
         conf = Config()
     base = ['qos']
     if not conf.exists(base):
         return None
 
     qos = conf.get_config_dict(base, key_mangling=('-', '_'),
                                get_first_key=True,
                                no_tag_node_value_mangle=True)
 
     for ifname in interfaces():
         if_node = Section.get_config_path(ifname)
 
         if not if_node:
             continue
 
         path = f'interfaces {if_node}'
         if conf.exists(f'{path} mirror') or conf.exists(f'{path} redirect'):
             type_node = path.split(" ")[1] # return only interface type node
             set_dependents(type_node, conf, ifname.split(".")[0])
 
     for policy in qos.get('policy', []):
         if policy in ['random_detect']:
             for rd_name in list(qos['policy'][policy]):
                 # There are eight precedence levels - ensure all are present
                 # to be filled later down with the appropriate default values
                 default_p_val = relative_defaults(
                     ['qos', 'policy', 'random-detect', rd_name, 'precedence'],
                     {'precedence': {'0': {}}},
                     get_first_key=True, recursive=True
                 )['0']
                 default_p_val = {key.replace('-', '_'): value for key, value in default_p_val.items()}
                 default_precedence = {
                     'precedence': {'0': default_p_val, '1': default_p_val,
                                    '2': default_p_val, '3': default_p_val,
                                    '4': default_p_val, '5': default_p_val,
                                    '6': default_p_val, '7': default_p_val}}
 
                 qos['policy']['random_detect'][rd_name] = dict_merge(
                     default_precedence, qos['policy']['random_detect'][rd_name])
 
     qos = conf.merge_defaults(qos, recursive=True)
 
+    if 'traffic_match_group' in qos:
+        for group, group_config in qos['traffic_match_group'].items():
+            if 'match_group' in group_config:
+                qos['traffic_match_group'][group]['match'] = _get_group_match(qos['traffic_match_group'], group)
+
     for policy in qos.get('policy', []):
         for p_name, p_config in qos['policy'][policy].items():
             # cleanup empty match config
             if 'class' in p_config:
                 for cls, cls_config in p_config['class'].items():
+                    if 'match_group' in cls_config:
+                        # merge group match to match
+                        for group in cls_config['match_group']:
+                            for match, match_conf in qos['traffic_match_group'].get(group, {'match': {}})['match'].items():
+                                if 'match' not in cls_config:
+                                    cls_config['match'] = dict()
+                                if match in cls_config['match']:
+                                    cls_config['match'][f'{group}-{match}'] = match_conf
+                                else:
+                                    cls_config['match'][match] = match_conf
+
                     if 'match' in cls_config:
                         cls_config['match'] = _clean_conf_dict(cls_config['match'])
                         if cls_config['match'] == {}:
                             del cls_config['match']
 
     return qos
 
+
+def _verify_match(cls_config: dict) -> None:
+    if 'match' in cls_config:
+        for match, match_config in cls_config['match'].items():
+            if {'ip', 'ipv6'} <= set(match_config):
+                raise ConfigError(
+                    f'Can not use both IPv6 and IPv4 in one match ({match})!')
+
+
+def _verify_match_group_exist(cls_config, qos):
+    if 'match_group' in cls_config:
+        for group in cls_config['match_group']:
+            if 'traffic_match_group' not in qos or group not in qos['traffic_match_group']:
+                Warning(f'Match group "{group}" does not exist!')
+
+
 def verify(qos):
     if not qos or 'interface' not in qos:
         return None
 
     # network policy emulator
     # reorder rerquires delay to be set
     if 'policy' in qos:
         for policy_type in qos['policy']:
             for policy, policy_config in qos['policy'][policy_type].items():
                 # a policy with it's given name is only allowed to exist once
                 # on the system. This is because an interface selects a policy
                 # for ingress/egress traffic, and thus there can only be one
                 # policy with a given name.
                 #
                 # We check if the policy name occurs more then once - error out
                 # if this is true
                 counter = 0
                 for _, path in dict_search_recursive(qos['policy'], policy):
                     counter += 1
                     if counter > 1:
                         raise ConfigError(f'Conflicting policy name "{policy}", already in use!')
 
                 if 'class' in policy_config:
                     for cls, cls_config in policy_config['class'].items():
                         # bandwidth is not mandatory for priority-queue - that is why this is on the exception list
                         if 'bandwidth' not in cls_config and policy_type not in ['priority_queue', 'round_robin', 'shaper_hfsc']:
                             raise ConfigError(f'Bandwidth must be defined for policy "{policy}" class "{cls}"!')
-                    if 'match' in cls_config:
-                        for match, match_config in cls_config['match'].items():
-                            if {'ip', 'ipv6'} <= set(match_config):
-                                 raise ConfigError(f'Can not use both IPv6 and IPv4 in one match ({match})!')
-
+                        _verify_match(cls_config)
+                        _verify_match_group_exist(cls_config, qos)
                 if policy_type in ['random_detect']:
                     if 'precedence' in policy_config:
                         for precedence, precedence_config in policy_config['precedence'].items():
                             max_tr = int(precedence_config['maximum_threshold'])
                             if {'maximum_threshold', 'minimum_threshold'} <= set(precedence_config):
                                 min_tr = int(precedence_config['minimum_threshold'])
                                 if min_tr >= max_tr:
                                     raise ConfigError(f'Policy "{policy}" uses min-threshold "{min_tr}" >= max-threshold "{max_tr}"!')
 
                             if {'maximum_threshold', 'queue_limit'} <= set(precedence_config):
                                 queue_lim = int(precedence_config['queue_limit'])
                                 if queue_lim < max_tr:
                                     raise ConfigError(f'Policy "{policy}" uses queue-limit "{queue_lim}" < max-threshold "{max_tr}"!')
                 if policy_type in ['priority_queue']:
                     if 'default' not in policy_config:
                         raise ConfigError(f'Policy {policy} misses "default" class!')
                 if 'default' in policy_config:
                     if 'bandwidth' not in policy_config['default'] and policy_type not in ['priority_queue', 'round_robin', 'shaper_hfsc']:
                         raise ConfigError('Bandwidth not defined for default traffic!')
 
     # we should check interface ingress/egress configuration after verifying that
     # the policy name is used only once - this makes the logic easier!
     for interface, interface_config in qos['interface'].items():
         for direction in ['egress', 'ingress']:
             # bail out early if shaper for given direction is not used at all
             if direction not in interface_config:
                 continue
 
             policy_name = interface_config[direction]
             if 'policy' not in qos or list(dict_search_recursive(qos['policy'], policy_name)) == []:
                 raise ConfigError(f'Selected QoS policy "{policy_name}" does not exist!')
 
             shaper_type, shaper_config = get_shaper(qos, interface_config, direction)
             tmp = shaper_type(interface).get_direction()
             if direction not in tmp:
                 raise ConfigError(f'Selected QoS policy on interface "{interface}" only supports "{tmp}"!')
 
+    if 'traffic_match_group' in qos:
+        for group, group_config in qos['traffic_match_group'].items():
+            _verify_match(group_config)
+            _verify_match_group_exist(group_config, qos)
+
     return None
 
+
 def generate(qos):
     if not qos or 'interface' not in qos:
         return None
 
     return None
 
 def apply(qos):
     # Always delete "old" shapers first
     for interface in interfaces():
         # Ignore errors (may have no qdisc)
         run(f'tc qdisc del dev {interface} parent ffff:')
         run(f'tc qdisc del dev {interface} root')
 
     call_dependents()
 
     if not qos or 'interface' not in qos:
         return None
 
     for interface, interface_config in qos['interface'].items():
         if not verify_interface_exists(interface, warning_only=True):
             # When shaper is bound to a dialup (e.g. PPPoE) interface it is
             # possible that it is yet not availbale when to QoS code runs.
             # Skip the configuration and inform the user via warning_only=True
             continue
 
         for direction in ['egress', 'ingress']:
             # bail out early if shaper for given direction is not used at all
             if direction not in interface_config:
                 continue
 
             shaper_type, shaper_config = get_shaper(qos, interface_config, direction)
             tmp = shaper_type(interface)
             tmp.update(shaper_config, direction)
 
     return None
 
+
 if __name__ == '__main__':
     try:
         c = get_config()
         verify(c)
         generate(c)
         apply(c)
     except ConfigError as e:
         print(e)
         exit(1)