Structural Tag Usage¶
The structural tag API aims to provide a JSON-config-based way to precisely describe the output format of an LLM. It is more flexible and dynamic than the OpenAI API:
Flexible: supports various structures, including tool calling, reasoning (<think>…</think>), etc.
Dynamic: allows a mixture of free-form text and structures such as tool calls, entering constrained generation when a pre-set trigger is met.
It can also be used in the LLM engine to implement the OpenAI Tool Calling API with strict format constraints, with these benefits:
Support the advanced tool calling features, such as forced tool calling, parallel tool calling, etc.
Support the tool calling format of most of the LLMs available in the market with minimal effort.
Usage¶
The structural tag is a response format. It’s compatible with the OpenAI API. With the structural tag, the request should be like:
{
"model": "...",
"messages": [
...
],
"response_format": {
"type": "structural_tag",
"format": {
"type": "...",
...
}
}
}
The format field requires a format object. We provide several basic format objects, and they can be composed to allow for more complex formats. Each format object represent a “chunk” of text.
Format Types¶
const_stringThe LLM output must exactly match the given string.
This is useful for like force reasoning, where the LLM output must start with “Let’s think step by step”.
{ "type": "const_string", "value": "..." }
json_schemaThe output should be a valid JSON object that matches the JSON schema.
style(optional, default:"json"): Controls how the content is parsed. Supported values:"json": Standard JSON format. The output must be valid JSON that conforms to the schema."qwen_xml": Qwen-coder XML style format. The output uses XML-style tags such as<parameter=name>value</parameter>to represent schema properties, as used in Qwen tool-calling formats. Use this when you need the same behavior as the legacyqwen_xml_parameterformat."minimax_xml": MiniMax XML style format. The output uses XML-style tags such as<parameter name="key">value</parameter>(attribute form) to represent schema properties, as used in MiniMax tool-calling formats. Same semantics asqwen_xmlexcept the tag format is<parameter name="key">instead of<parameter=key>."deepseek_xml": DeepSeek-v3.2 XML style format. The output should be in the format of<{dsml_token}parameter name=\"key\" string=\"true|false\">value</{dsml_token}parameter>.
When
styleis omitted, it defaults to"json".{ "type": "json_schema", "json_schema": { ... }, "style": "json" }For Qwen / MiniMax / Deepseek XML style output, set
styletoqwen_xml/minimax_xml/deepseek_xml:{ "type": "json_schema", "json_schema": { ... }, "style": "qwen_xml" / "minimax_xml" / "deepseek_xml" }grammarThis format can be used to match a given ebnf grammar.
{ "type": "grammar", "grammar": "..." }
We can use it as the context of other structural tags as well. When using grammar constraints, you need to be especially careful. If the grammar is too general (for example .*), it will cause the subsequent constraints to become ineffective.
regexThis format can be used to match a given ebnf grammar.
{ "type": "regex", "pattern": "..." }
We can use it as the context of other structural tags as well. As GrammarFormat, if the regex pattern is too general, it will cause the subsequent constraints to become inefficient as well.
sequenceThe output should match a sequence of elements.
{ "type": "sequence", "elements": [ { "type": "...", }, { "type": "...", }, ... ] }orThe output should follow any of the elements.
{ "type": "or", "elements": [ { "type": "...", }, { "type": "...", }, ... ] }tagThe output must follow
begin content end.beginandendcan be strings ortokenformat objects, andcontentcan be any format object. This is useful for LLM outputs such as<think>...</think>or<function>...</function>.{ "type": "tag", "begin": "...", "content": { "type": "...", }, "end": "..." }
beginandendcan also betokenformat objects for token-level boundary matching:{ "type": "tag", "begin": {"type": "token", "token": "<|tool_call|>"}, "content": {"type": "json_schema", "json_schema": ...}, "end": {"type": "token", "token": "<|end|>"} }any_textThe any_text format allows any text.
{ "type": "any_text", "excludes": ["...", ] }
We will handle it as a special case when wrapped in a tag:
{ "type": "tag", "begin": "...", "content": { "type": "any_text", "excludes": ["...", ] }, "end": "...", }
It first accepts the begin tag (can be empty), then any text except the end tag and the excludes, then the end tag.
triggered_tagsThe output will match triggered tags. It can allow any output until a trigger is encountered, then dispatch to the corresponding tag; when the end tag is encountered, the grammar will allow any following output, until the next trigger is encountered.
Each tag should be matched by exactly one trigger. “matching” means the trigger should be a prefix of the begin tag. All the strings in
excludeswill not be accepted before the tag is triggerred.Note: Tags inside
triggered_tagsmust use stringbeginfields (nottokenformat objects). For token-level dispatch, usetoken_triggered_tagsinstead.{ "type": "triggered_tags", "triggers": ["<function="], "tags": [ { "begin": "...", "content": { ... }, "end": "..." }, { "begin": "...", "content": { ... }, "end": "..." }, ], "at_least_one": bool, "stop_after_first": bool, "excludes": ["...", ] }For example,
{ "type": "triggered_tags", "triggers": ["<function="], "tags": [ { "begin": "<function=func1>", "content": { "type": "json_schema", "json_schema": ... }, "end": "</function>", }, { "begin": "<function=func2>", "content": { "type": "json_schema", "json_schema": ... }, "end": "</function>", }, ], "at_least_one": false, "stop_after_first": false, }The above structural tag can accept the following outputs:
<function=func1>{"name": "John", "age": 30}</function> <function=func2>{"name": "Jane", "age": 25}</function> any_text<function=func1>{"name": "John", "age": 30}</function>any_text1<function=func2>{"name": "Jane", "age": 25}</function>any_text2at_least_onemakes sure at least one of the tags must be generated. The first tag will be generated at the beginning of the output.stop_after_firstwill reach the end of thetriggered_tagsstructure after the first tag is generated. If there are following tags, they will still be generated; otherwise, the generation will stop.tags_with_separatorThe output should match zero, one, or more tags, separated by the separator, with no other text allowed.
{ "type": "tags_with_separator", "tags": [ { "type": "tag", "begin": "...", "content": { "type": "...", }, "end": "...", }, ], "separator": "...", "at_least_one": bool, "stop_after_first": bool, }For example,
{ "type": "tags_with_separator", "tags": [ { "type": "tag", "begin": "<function=func1>", "content": { "type": "json_schema", "json_schema": ... }, "end": "</function>", }, ], "separator": ",", "at_least_one": false, "stop_after_first": false, }The above structural tag can accept an empty string, or the following outputs:
<function=func1>{"name": "John", "age": 30}</function> <function=func1>{"name": "John", "age": 30}</function>,<function=func2>{"name": "Jane", "age": 25}</function> <function=func1>{"name": "John", "age": 30}</function>,<function=func2>{"name": "Jane", "age": 25}</function>,<function=func1>{"name": "John", "age": 30}</function>at_least_onemakes sure at least one of the tags must be generated.stop_after_firstwill reach the end of thetags_with_separatorstructure after the first tag is generated. If there are following tags, they will still be generated; otherwise, the generation will stop.optionalThe inner format may appear 0 or 1 time (EBNF optional). The output can either match the content once or be empty.
content(required): A format object. It can be any of the format types.{ "type": "optional", "content": { "type": "..." } }
For example, to allow an optional prefix:
{ "type": "optional", "content": { "type": "const_string", "value": "Optional prefix: " } }
The above accepts either an empty string or exactly
Optional prefix:.plusThe inner format must appear 1 or more times (EBNF plus). The output matches one or more consecutive occurrences of the content.
content(required): A format object. It can be any of the format types.{ "type": "plus", "content": { "type": "..." } }
For example, to require one or more comma-separated items (simplified):
{ "type": "plus", "content": { "type": "const_string", "value": "item" } }
The above accepts
item,itemitem,itemitemitem, and so on.starThe inner format may appear 0 or more times (EBNF star). The output matches zero or more consecutive occurrences of the content.
content(required): A format object. It can be any of the format types.{ "type": "star", "content": { "type": "..." } }
For example, to allow zero or more occurrences of a fragment:
{ "type": "star", "content": { "type": "const_string", "value": "x" } }
The above accepts an empty string, or
x,xx,xxx, and so on.repeatThe inner format may appear between
minandmaxtimes (inclusive). This generalizesplus(min=1, unbounded) andstar(min=0, unbounded). Usemax: -1for an unbounded upper limit (i.e. “at leastmintimes”).min(required): Minimum number of occurrences (inclusive). Must be non-negative.max(required): Maximum number of occurrences (inclusive). Use-1for unbounded.content(required): A format object. It can be any of the format types.{ "type": "repeat", "min": 1, "max": 3, "content": { "type": "const_string", "value": "item" } }
The above accepts
item,itemitem, oritemitemitem(exactly 1 to 3 times).For “at least N times” (no upper bound), set
maxto-1:{ "type": "repeat", "min": 2, "max": -1, "content": { "type": "const_string", "value": "x" } }
The above accepts
xx,xxx,xxxx, and so on (2 or more times).QwenXMLParameterFormat(not recommended)Deprecated. This format is kept for backward compatibility only. Prefer using
json_schemawithstyle: "qwen_xml"instead (see thejson_schemaformat above).This is designed for the parameter format of Qwen3-coder. The output should match the given JSON schema in XML style.
{ "type": "qwen_xml_parameter", "json_schema": { ... } }For example,
{ "type": "qwen_xml_parameter", "json_schema": { "type": "object", "properties": {"name": {"type": "string"}, "age": {"type": "integer"}}, "required": ["name", "age"], }, }
This can accept outputs such like:
<parameter=name>Bob</parameter><parameter=age>\t100\n</parameter> <parameter=name>Bob</parameter>\t\n<parameter=age>\t100\n</parameter> <parameter=name>Bob</parameter><parameter=age>100</parameter> <parameter=name>"Bob<"</parameter><parameter=age>100</parameter>
Note that strings here are in XML style. Moreover, if the parameter’s type is
object, the innerobjectwill still be in JSON style. For example:{ "type": "qwen_xml_parameter", "json_schema": { "type": "object", "properties": { "address": { "type": "object", "properties": {"street": {"type": "string"}, "city": {"type": "string"}}, "required": ["street", "city"], } }, "required": ["address"], }, }
These are valid outputs:
<parameter=address>{"street": "Main St", "city": "New York"}</parameter> <parameter=address>{"street": "Main St", "city": "No more xml escape&<>"}</parameter>And this is an invalid output:
<parameter=address><parameter=street>Main St</parameter><parameter=city>New York</parameter></parameter>
tokenMatches a single token by token ID or token string. Can be used standalone or as the
begin/endof atag.{"type": "token", "token": 42} {"type": "token", "token": "<|tool_call|>"}
When
tokenis a string, it is resolved to a token ID viatokenizer_info.exclude_tokenMatches any single token except those in the given set. When wrapped in a
tagwhoseendis atokenformat, the end token is automatically excluded.{"type": "exclude_token"} {"type": "exclude_token", "exclude_tokens": [42, "</s>"]}
Elements in
exclude_tokenscan be integers (token IDs) or strings (token strings resolved viatokenizer_info).any_tokensMatches zero or more tokens, excluding those in the given set. Semantically equivalent to
star(exclude_token(...)). When wrapped in atagwhoseendis atokenformat, the end token is automatically excluded.{"type": "any_tokens"} {"type": "any_tokens", "exclude_tokens": [42, "</s>"]}
Elements in
exclude_tokenscan be integers (token IDs) or strings (token strings resolved viatokenizer_info).token_triggered_tagsSimilar to
triggered_tags, but triggers and excludes operate at the token level. When a trigger token is generated, the output dispatches to the corresponding tag.Note: Tags inside
token_triggered_tagsmust usetokenformat objects as theirbeginfields (not strings). For string-level dispatch, usetriggered_tagsinstead.{ "type": "token_triggered_tags", "trigger_tokens": [100, "<|tool_call|>"], "tags": [ { "begin": {"type": "token", "token": 100}, "content": {"type": "json_schema", "json_schema": ...}, "end": {"type": "token", "token": 101} } ], "exclude_tokens": ["</s>"], "at_least_one": false, "stop_after_first": false }trigger_tokensandexclude_tokenselements can be integers or strings.at_least_oneandstop_after_firsthave the same semantics as intriggered_tags.dispatchAccepts any string except the excluded string. But when any of the case strings is generated, the following format must follow the corresponding format. When the loop is false, after generating the format, we reach the end of the format. Otherwise, it will continue to allow generating any string and match the next case.
rules(required): Non-empty array of 2-element JSON arrays. Each entry is[pattern, content_format]: the pattern is a string, followed by a format object for the content after that pattern.loop(optional, defaulttrue): If true, after handling one dispatched format, it will continue to allow free-form text and match the next pattern. Otherwise, the matching of this format ends after handling the first dispatched format.exclude_tokens(optional, default[]): List of strings that must not appear in the free-form text.Corresponds to an Aho-Corasick automaton under the hood to allow efficient and accurate pattern matching in a free-formed string.
Common usage: Put the end string inside the exclude, and put another end string after it. E.g.
format: sequence sequence: [ DispatchFormat(cases=..., excludes=["abc"]) ConstStringFormat("abc") ]This means when the LLM generates abc, the matching of the first DispatchFormat ends and gets into the following ConstStringFormat. You can also use the “end” part of the TagFormat to achieve the same goal.
token_dispatchSimilar to
dispatchtype, while the patterns are token IDs or token strings.rules(required): Non-empty array of 2-element JSON arrays. Each entry is[pattern_token, content_format]: the pattern is a token ID (integer) or token string (resolved viatokenizer_info), followed by a format object for the content after that token.loop(optional, defaulttrue): If true, after one dispatched format, it will continue to allow free-form text and match the next pattern. Otherwise, the matching of this format ends after handling the first dispatched format.exclude_tokens(optional, default[]): List of Tokens that must not appear in the free-form text.{ "type": "token_dispatch", "cases": [ [100, {"type": "const_string", "value": "x"}], ["<|tool|>", {"type": "json_schema", "json_schema": {...}}] ], "loop": true, "exclude_tokens": ["</s>"] }
Examples¶
Example 1: Tool calling¶
The structural tag can support most common tool calling formats.
Llama JSON-based tool calling, Gemma:
{"name": "function_name", "parameters": params}
Corresponding structural tag:
{
"type": "structural_tag",
"format": {
"type": "triggered_tags",
"triggers": ["{\"name\":"],
"tags": [
{
"begin": "{\"name\": \"func1\", \"parameters\": ",
"content": {"type": "json_schema", "json_schema": ...},
"end": "}"
},
{
"begin": "{\"name\": \"func2\", \"parameters\": ",
"content": {"type": "json_schema", "json_schema": ...},
"end": "}"
},
],
},
}
Llama user-defined custom tool calling:
<function=function_name>params</function>
Corresponding structural tag:
{
"type": "structural_tag",
"format": {
"type": "triggered_tags",
"triggers": ["<function="],
"tags": [
{
"begin": "<function=func1>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
{
"begin": "<function=func2>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
],
},
}
Qwen 2.5/3, Hermes:
<tool_call>
{"name": "get_current_temperature", "arguments": {"location": "San Francisco, CA, USA"}}
</tool_call>
Corresponding structural tag:
{
"type": "structural_tag",
"format": {
"type": "triggered_tags",
"triggers": ["<tool_call>"],
"tags": [
{
"begin": "<tool_call>\n{\"name\": \"func1\", \"arguments\": ",
"content": {"type": "json_schema", "json_schema": ...},
"end": "}\n</tool_call>",
},
{
"begin": "<tool_call>\n{\"name\": \"func2\", \"arguments\": ",
"content": {"type": "json_schema", "json_schema": ...},
"end": "}\n</tool_call>",
},
],
},
}
DeepSeek:
There is a special tag <|tool▁calls▁begin|> ... <|tool▁calls▁end|> quotes the whole tool calling part.
<|tool▁calls▁begin|><|tool▁call▁begin|>function<|tool▁sep|>function_name_1
```jsonc
{params}
```<|tool▁call▁end|>
```jsonc
{params}
```<|tool▁call▁end|><|tool▁calls▁end|>
Corresponding structural tag:
{
"type": "structural_tag",
"format": {
"type": "triggered_tags",
"triggers": ["<|tool▁calls▁begin|>"],
"tags": [
{
"begin": "<|tool▁calls▁begin|>",
"end": "<|tool▁calls▁end|>",
"content": {
"type": "tags_with_separator",
"separator": "\n",
"tags": [
{
"begin": "<|tool▁call▁begin|>function<|tool▁sep|>function_name_1\n```jsonc\n",
"content": {"type": "json_schema", "json_schema": ...},
"end": "\n```<|tool▁call▁end|>",
},
{
"begin": "<|tool▁call▁begin|>function<|tool▁sep|>function_name_2\n```jsonc\n",
"content": {"type": "json_schema", "json_schema": ...},
"end": "\n```<|tool▁call▁end|>",
}
]
}
}
],
"stop_after_first": true,
},
}
Phi-4-mini:
Similar to DeepSeek-V3, but the tool calling part is wrapped in <|tool_call|>...<|/tool_call|> and organized in a list.
<|tool_call|>[{"name": "function_name_1", "arguments": params}, {"name": "function_name_2", "arguments": params}]<|/tool_call|>
Corresponding structural tag:
{
"type": "structural_tag",
"format": {
"type": "triggered_tags",
"triggers": ["<|tool_call|>"],
"tags": [
{
"begin": "<|tool_call|>[",
"end": "]<|/tool_call|>",
"content": {
"type": "tags_with_separator",
"separator": ", ",
"tags": [
{
"begin": "{\"name\": \"function_name_1\", \"arguments\": ",
"content": {"type": "json_schema", "json_schema": ...},
"end": "}",
},
{
"begin": "{\"name\": \"function_name_2\", \"arguments\": ",
"content": {"type": "json_schema", "json_schema": ...},
"end": "}",
}
]
}
}
],
"stop_after_first": true,
},
}
Example 2: Force think¶
The output should start with a reasoning part (<think>...</think>), then can generate a mix of text and tool calls.
Format:
<think> any_text </think> any_text <function=func1> params </function> any_text
Corresponding structural tag:
{
"type": "structural_tag",
"format": {
"type": "sequence",
"elements": [
{
"type": "tag",
"begin": "<think>",
"content": {"type": "any_text"},
"end": "</think>",
},
{
"type": "triggered_tags",
"triggers": ["<function="],
"tags": [
{
"begin": "<function=func1>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
{
"begin": "<function=func2>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
],
},
],
},
}
Example 3: Think & Force tool calling (Llama style)¶
The output should start with a reasoning part (<think>...</think>), then need to generate exactly one tool call in the tool set.
Format:
<think> any_text </think> <function=func1> params </function>
Corresponding structural tag:
{
"type": "structural_tag",
"format": {
"type": "sequence",
"elements": [
{
"type": "tag",
"begin": "<think>",
"content": {"type": "any_text"},
"end": "</think>",
},
{
"type": "triggered_tags",
"triggers": ["<function="],
"tags": [
{
"begin": "<function=func1>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
{
"begin": "<function=func2>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
],
"stop_after_first": true,
"at_least_one": true,
},
],
},
}
Example 4: Think & force tool calling (DeepSeek style)¶
The output should start with a reasoning part (<think>...</think>), then must generate a tool call following the DeepSeek style.
Config:
{
"type": "structural_tag",
"format": {
"type": "sequence",
"elements": [
{
"type": "tag",
"begin": "<think>",
"content": {"type": "any_text"},
"end": "</think>",
},
{
"type": "tag_and_text",
"triggers": ["<|tool▁calls▁begin|>"],
"tags": [
{
"begin": "<|tool▁calls▁begin|>",
"end": "<|tool▁calls▁end|>",
"content": {
"type": "tags_with_separator",
"separator": "\n",
"tags": [
{
"begin": "<|tool▁call▁begin|>function<|tool▁sep|>function_name_1\n```jsonc\n",
"content": {"type": "json_schema", "json_schema": ...},
"end": "\n```<|tool▁call▁end|>",
},
{
"begin": "<|tool▁call▁begin|>function<|tool▁sep|>function_name_2\n```jsonc\n",
"content": {"type": "json_schema", "json_schema": ...},
"end": "\n```<|tool▁call▁end|>",
}
],
"at_least_one": true, // Note this line!
"stop_after_first": true, // Note this line!
}
}
],
"stop_after_first": true,
},
],
},
},
Example 5: Force non-thinking mode¶
Qwen-3 has a hybrid thinking mode that allows switching between thinking and non-thinking mode. Thinking mode is the same as above, while in non-thinking mode, the output would start with a empty thinking part <think></think>, and then can generate any text.
We now specify the non-thinking mode.
{
"type": "structural_tag",
"format": {
"type": "sequence",
"elements": [
{
"type": "const_string",
"text": "<think></think>"
},
{
"type": "triggered_tags",
"triggers": ["<tool_call>"],
"tags": [
{
"begin": "<tool_call>\n{\"name\": \"func1\", \"arguments\": ",
"content": {"type": "json_schema", "json_schema": ...},
"end": "}\n</tool_call>",
},
{
"begin": "<tool_call>\n{\"name\": \"func2\", \"arguments\": ",
"content": {"type": "json_schema", "json_schema": ...},
"end": "}\n</tool_call>",
},
],
},
],
},
}
Example 6: Token-level tool calling¶
Some models use special tokens (not byte strings) to delimit tool calls. The token_triggered_tags format handles this by dispatching on token IDs.
{
"type": "structural_tag",
"format": {
"type": "sequence",
"elements": [
{
"type": "tag",
"begin": {"type": "token", "token": "<|think_start|>"},
"content": {"type": "any_tokens"},
"end": {"type": "token", "token": "<|think_end|>"}
},
{
"type": "token_triggered_tags",
"trigger_tokens": ["<|tool_call_start|>"],
"tags": [
{
"begin": {"type": "token", "token": "<|tool_call_start|>"},
"content": {"type": "json_schema", "json_schema": ...},
"end": {"type": "token", "token": "<|tool_call_end|>"}
}
],
"exclude_tokens": ["<|eos|>"],
"at_least_one": true
}
]
}
}
Compatibility with the OpenAI Tool Calling API¶
The structural tag can be used to implement the OpenAI Tool Calling API with strict format
constraints. In LLM serving engines, you can use the xgrammar Python package to construct the
structural tag and apply it to constrained decoding.
In the OpenAI Tool Calling API, a set of tools is provided using JSON schema. There are also several features: tool choice (control at least one tool or exactly one tool is called), parallel tool calling (allow only one tool or multiple tools can be called in one round), etc.
You can construct the structural tag according to the provided tools, and the LLM’s specific tool calling format. The structural tag can be used in XGrammar’s constrained decoding workflow to enable strict format constraints.
Tool Choice¶
tool_choice is a parameter in the OpenAI API. It can be
auto: Let the model decide which tool to userequired: Call at least one tool in the tool set{"type": "function", "function": {"name": "function_name"}}: The forced mode, call exactly one specific function
The required mode can be implemented by
{
"type": "structural_tag",
"format": {
"type": "triggered_tags",
"triggers": ["<function="],
"tags": [
{
"begin": "<function=func1>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
{
"begin": "<function=func2>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
],
"at_least_one": true,
},
}
The forced mode can be implemented by
{
"type": "structural_tag",
"format": {
"type": "tag",
"begin": "<function=func1>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
}
Parallel Tool Calling¶
OAI’s parallel_tool_calls parameter controls if the model can call multiple functions in one round.
If
true, the model can call multiple functions in one round. (This is default)If
false, the model can call at most one function in one round.
triggered_tags and tags_with_separator has a parameter stop_after_first to control if the
generation should stop after the first tag is generated. So the false mode can be implemented by:
{
"type": "structural_tag",
"format": {
"type": "triggered_tags",
"triggers": ["<function="],
"tags": [
{
"begin": "<function=func1>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
{
"begin": "<function=func2>",
"content": {"type": "json_schema", "json_schema": ...},
"end": "</function>",
},
],
"stop_after_first": true,
},
}
The true mode can be implemented by setting stop_after_first to false.
Next Steps¶
For API reference, see Structural Tag API Reference.
For advanced usage, see Advanced Topics of the Structural Tag.