Copyright © 2012-2014 Alexey Sokolov <email@example.com>
Copyright © 2012 Stéphan Kochen <firstname.lastname@example.org>
Copyright © 2012 Kyle Fuller <email@example.com>
Copyright © 2016 Kiyoshi Aman <firstname.lastname@example.org>
Copyright © 2016-2019 James Wheare <email@example.com>
Unlimited redistribution and modification of this document is allowed provided that the above copyright notice and this permission notice remains intact.
Message tags are a mechanism for adding additional metadata on a per-message basis. This is achieved via an extension to the protocol message format, enabled via capability negotiation.
Tags are simple keys that MAY have optional string data as values.
Tagged messages can be sent by both servers and clients. The usage of individual tags are specified in their own documents.
The message pseudo-BNF, as defined in RFC 1459, section 2.3.1 is extended as follows:
<message> ::= ['@' <tags> <SPACE>] [':' <prefix> <SPACE> ] <command> [params] <crlf> <tags> ::= <tag> [';' <tag>]* <tag> ::= <key> ['=' <escaped_value>] <key> ::= [ <client_prefix> ] [ <vendor> '/' ] <key_name> <client_prefix> ::= '+' <key_name> ::= <non-empty sequence of ascii letters, digits, hyphens ('-')> <escaped_value> ::= <sequence of zero or more utf8 characters except NUL, CR, LF, semicolon (`;`) and SPACE> <vendor> ::= <host>
The ordering of tags is not meaningful.
Individual tag keys MUST only be used a maximum of once per message. Implementations receiving messages with more than one occurrence of a tag key name SHOULD disregard all but the final occurrence.
Implementations MUST treat tag key names as case-sensitive opaque identifiers and MUST NOT perform any validation that would reject the message if an invalid tag key name is used. This allows future modifications to the tag key name format.
Tag values MUST be encoded as UTF8. This ensures a shared interoperable baseline for data exchange. If tag values are encountered that cannot be decoded as UTF8, implementations MAY drop the value entirely but SHOULD NOT substitute replacement bytes in place of invalid data, which can result in collisions.
Implementations MUST interpret empty tag values (e.g.
foo=) as equivalent to missing tag values (e.g.
foo). Specifications MUST NOT differentiate meaning between tags with empty and missing values. Implementations MAY normalise tag values by converting the empty form to the missing form, but MUST NOT convert values from missing to empty, to prevent size limit issues.
The mapping between characters in tag values and their representation in
<escaped value> is defined as follows:
|all others||the character itself|
This escape format is space efficient, and ensures that message parts can easily be split on spaces and semi-colons before further parsing.
If a lone
\ exists at the end of an escaped value (with no escape character following it), then there
SHOULD be no output character. For example, the escaped value
test\ should unescape to
test. If a
\ exists with no valid escape character (for example,
\b), then the invalid backslash SHOULD be
dropped. For example,
\b should unescape to just
Tags are enabled via capability negotiation. Clients and servers that negotiate a capability that uses tags MUST support the full tag format above.
The following capabilities implicitly depend on the tag format:
But there is also a generic capability for negotiating tag usage.
message-tags capability allows clients to indicate support for
receiving any well-formed tag, whether or not it is recognised or used. It also frees servers from having to filter individual message tags for each client response.
message-tags capability is the only guarantee that an implementation supports the full tag format. Following the robustness principle, implementations that use tags MUST support the full message tag format. However, implementations SHOULD NOT assume support for all tags when an individual tag capability is negotiated.
This capability enables the use of client-only tags and the
TAGMSG command, described below.
Clients requesting this capability indicate that they are capable of parsing all well-formed tags, even if they don’t handle them specifically.
Servers advertising this capability indicate that they are capable of parsing any tag received from a client, with or without the client-only prefix.
Client-only tags are tags normally sent by clients that servers MUST attach as-is
to any relevant message relayed to other clients. A client-only tag is prefixed
with a plus sign (
+). Servers MAY send client-only tags that weren’t provided explicitly by the client.
The client-only tag prefix allows servers to safely relay untrusted client tags, keeping them distinct from unprefixed tags that carry verified meaning.
Client-only tags MUST be relayed on
TAGMSG messages, and MAY be relayed on other messages.
Any server-sent tags attached to messages MUST be included before client-only tags to prevent them from being pushed outside of the byte limit.
The expected client behaviour of individual client-only tags are defined in separate specifications.
This means client-only tags that aren’t specified in the IRCv3 extension registry MUST use a vendor prefix and SHOULD be submitted to the IRCv3 working group for consideration if they are appropriate for more widespread adoption. See Rules for naming message tags.
Client-only tags are intended to replace the use of future CTCP commands.
Servers MAY apply moderation to client-only tags using existing or newly specified modes or configuration. See the RPL_ISUPPORT Tokens section for further information.
TAGMSGtag-only message 🔗
Command: TAGMSG Parameters: <msgtarget>
A new message command
TAGMSG is defined for sending messages with tags but no text content.
This message MUST be delivered to targets in the same way as
messages. This means for example, honouring channel membership, modes,
Servers MUST NOT deliver
TAGMSG to clients that haven’t negotiated the message tags capability.
PRIVMSG in RFC2812 for more details on replies and examples.
Clients that receive a
TAGMSG command MUST NOT display them in the message history by default. Display guidelines are defined in the specifications of tags attached to the message.
Full tag names, including any vendor prefixes MUST be treated as an opaque identifier.
There are two tag namespaces:
Names which contain a slash character (
/) designate a vendor-specific tag namespace.
These names are prefixed by a valid DNS domain name.
In cases where the prefix contains non-ASCII characters, punycode MUST be used,
Vendor-Specific tags should be submitted to the IRCv3 working group for consideration.
draft/ vendor namespace may be used when the working group is considering tag specifications.
However, vendor names should be preferred.
While tags are in draft status, they may need to be given a new identifier, to prevent
implementation compatibility issues. When updating a draft tag key name, the
typical method is to add
-0.x to the name, where
x is a version number. For example:
draft/foo would become
draft/foo-0.2, and so on.
Reserved names for which a corresponding document exists in the IRCv3 Extension Registry.
The IRCv3 Working Group reserves the right to reuse names which have not been submitted to the registry. If you do not wish to submit your tag then you MUST use a vendor-specific name (see above).
The size limit for message tags is 8191 bytes, including the leading
'@' (0x40) and trailing space
' ' (0x20) characters. The size limit for the rest of the message remains unchanged at 512 bytes.
This limit is separated between tags added by the server and tags sent by the client. This prevents servers from overflowing the overall limit by adding tags to a client message sent within the allowed limit.
In the following description, tag data describes the bytes between the leading and trailing tag separators mentioned above.
Clients MUST NOT send messages with tag data exceeding 4094 bytes, this includes tags with or without the client-only prefix.
Servers MUST NOT add tag data exceeding 4094 bytes to messages.
<server_max> (4096) :: '@' <tag_data 4094> ' ' <client_max> (4096) :: '@' <tag_data 4094> ' ' <combined_max> (8191) :: '@' <tag_data 4094> ';' <tag_data 4094> ' '
Servers MUST reply with the
417) error numeric if a client sends a message with more tag data than the allowed limit. Servers MUST NOT truncate tags and MUST always reject lines that exceed this limit.
417 ERR_INPUTTOOLONG ":Input line was too long"
If a server sends a message with more tag data than the allowed limit, clients MAY ignore the message.
This specification defines the optional
CLIENTTAGDENY token for use in
RPL_ISUPPORT (005) responses.
Servers SHOULD use this token to communicate to clients that certain client-only tags are blocked and will be silently ignored. Blocking client-only tags is not a recommended default behaviour, but server administrators might wish to do so for moderation reasons.
This token allows clients to selectively remove features from their user interface that rely on any client-only tag that the server has blocked.
Note that blocking tags will not necessarily block the message that the tag is attached to.
Clients MAY still send blocked tags to the server.
CLIENTTAGDENY token value is a comma
, (0x2C) separated list of blocked client-only tags. The client-only prefix (
+) is omitted when a tag appears in this list.
* (0x2A) indicates that all client-only tags are blocked. When used, this MUST be the first item in the list.
- (0x2D) indicates that a block is negated, i.e. when certain client-only tags are exempt from a catch-all block.
An empty or missing
CLIENTTAGDENY matches the default case and indicates that all client-only tags are allowed.
An example where all client-only tags are allowed (omitting the token entirely is recommended instead)
An example where all client-only tags are blocked
An example where the
+example/bar client-only tags are allowed but all others are blocked
An example where only the
+example/bar client-only tags are blocked
Client-only tags should be treated as untrusted data. They can contain any value and clients cannot assume servers validate them in any way. The server MAY unescape tag values after receiving data and escape those values before sending them out.
Tags without the client-only prefix MUST be removed by the server before being relayed with any message to another client.
Some specifications may involve servers accepting client-sent tags without the client-only prefix. These specifications MUST define a process to be performed by the server on these tags prior to their removal.
There is no guarantee that other clients will support or display the client-only tags they receive, so these tags should only be used to enhance messages with non-critical information. Messages should still make sense to clients with no support for tags.
This section is non-normative.
Moderation tools available to channel and network operators typically operate on message text and sender information only. To avoid abusive content being sent in client-only tags, servers, services and management bot implementations may wish to enable moderation features that act on tag content as well.
This section is non-normative. The tags used in these examples may or may not have a specified meaning elsewhere.
A message with 0 tags:
:firstname.lastname@example.org PRIVMSG me :Hello
A message sent by the server with 3 tags:
S: @aaa=bbb;ccc;example.com/ddd=eee :email@example.com PRIVMSG me :Hello
aaa with value
ccc with no value
ddd specific to software of
example.com with value
A message sent by a client with the
C: @example-tag=example-value PRIVMSG #channel :Message
A message sent by a client with the
+example-client-tag client-only tag:
C: @+example-client-tag=example-value PRIVMSG #channel :Message
In this example
firstname.lastname@example.org a URL in a channel (without tags)
email@example.com with the URL title in the message body and the favicon URL included as the value of the
The server sends these messages:
S: :firstname.lastname@example.org PRIVMSG #channel :https://example.com/a-news-story S: @+icon=https://example.com/favicon.png :email@example.com PRIVMSG #channel :Example.com: A News Story
An example of a vendor-prefixed client-only tag:
C: @+example.com/foo=bar :irc.example.com NOTICE #channel :A vendor-prefixed client-only tagged message
A client-only tag
+example with a value containing valid raw and escaped characters:
In this example, plus signs, colons, equals signs and commas are transmitted raw in tag values; while semicolons, spaces and backslashes are escaped.
C: @+example=raw+:=,escaped\:\s\\ :irc.example.com NOTICE #channel :Message
TAGMSG sent by a client with the
+example-client-tag client-only tag:
C: @+example-client-tag=example-value TAGMSG #channel
TAGMSG sent by a client to channel ops with a client-only tag and
C: @+example-client-tag=example-value TAGMSG @#channel
TAGMSG sent to a channel with
labeled-response and client-only tags. The message is returned by the server with a
msgid tag via labeled
echo-message to the originating client and as normal to other clients in the channel.
C1-S: @label=123;+example-client-tag=example-value TAGMSG #channel S-C1: @label=123;msgid=abc;+example-client-tag=example-value :firstname.lastname@example.org TAGMSG #channel S-C2: @msgid=abc;+example-client-tag=example-value :email@example.com TAGMSG #channel
TAGMSG sent by a client with tags that exceed the size limit. The message is rejected by the server with an
417) error numeric.
[...] is used to represent tags omitted for readability.
C: @+tag1;+tag2;+tag[...];+tag5000 TAGMSG #channel S: :server.example.com 417 nick :Input line was too long
TAGMSG sent by a client with an un-prefixed tag that has no specified behaviour. The server removes the tag before relaying the message
C: @unknown-tag TAGMSG #channel S: :firstname.lastname@example.org TAGMSG #channel
Previous versions of this spec did not explicitly state that tag names are case-sensitive, though it was implied by their “opacity”
Previous versions of this spec did not specify that the full tag name MUST be parsed as an opaque identifier. This was added to improve client resiliency.
Previous versions of this spec did not specify how to handle trailing backslashes with no escape character. This was added to help consistency across implementations.
Previous versions of this spec did not specify how to handle invalid escapes. This was clarified to help consistency across implementations.
Previous versions of this spec did not specify the difference between empty and missing tag values.
Previous versions of this spec did not specify the UTF8 encoding for tag values
Previous versions of this spec did not define the
RPL_ISUPPORT token and were unclear about moderation allowances for client-only tags.
Software supporting message-tags: Ergo, IRCCloud Teams, InspIRCd, UnrealIRCd, AdiIRC, Ambassador, catgirl, Irssi, mIRC, Quassel, senpai, WeeChat, gamja, IRCCloud, Kiwi IRC, The Lounge, PIRC.pl web client, CoreIRC, Quasseldroid, Goguma, IRCCloud (as Server), KiwiBNC (as Server), KiwiBNC (as Client), pounce (as Server), pounce (as Client), soju (as Server), soju (as Client), BitBot, Eggdrop, Limnoria, Moon Moon, girc, irc-framework, ircrobots, Kitteh IRC Client Library, pydle