Queries are a powerful tool for enabling your players to find the most relevant opponents when using the Matchmaker and the most suitable multiplayer matches when listing available matches. A query expression defines the user’s criteria for finding their opponents or available matches - think of it as a big AND statement - with the query then inspecting the Matchmaker properties or match label, respectively, to deliver the results.
Querying the match label is only possible if the label is a JSON value.
Queries are composed of one or more query terms using the syntax field:value. A query can contain any number of terms with each term separated by a space, for example field:value field1:value1.
The same syntax can be used for any value type, for example:
Strings: region:europe
Numbers: level:10
Numeric Ranges: rank:<=5
Date Ranges: created:>"2021-12-25"
See the Operators section below to learn more about writing query terms, and Boosting for details about ordering results based on the weight given to individual query terms.
The standard syntax matches the whole value, meaning there must be equality between the query value and returned result, not just similarity. For example, a query of mode:free would not return matches with a mode of freeforall.
Queries also inspect individual elements inside an array for a match. For example, an indexed value: {"field": [5, 10, 15]} will be returned as a match for any query including a term field:5, field:10, or field:15.
Should is the default operator used for query terms. In practice it means that the results should have the given value, but it is not a strict requirement.
For example, if you are searching for opponents using a query of region:europe the results will include good results - meaning opponents in the Europe region - if available but will also include opponents from any other region if not enough good results can be found.
The MUST operator, indicated by a + at the start of a query term, enforces a string requirement of the given value. Using the same example as above but with the MUST operator (+region:europe) means that only opponents from the Europe region will be returned in the result and, if not enough are available, no result is returned.
The MUST NOT operator is indicated by a - at the start of the query term. It enforces a string requirement for the exclusion of the query term, i.e. any returned result will absolutely not have that value.
For example, if a user doesn’t want to see any matches with a level above 10, the query term could look like -level:>10.
Queries can contain both numeric and date ranges as terms.
Numeric ranges can be used in query terms with the >, >=, <, and <= operators. These same operators can be used for date ranges with the distinction that the date be in quotes, for example created:>"2022-01-01" to return results created after January 1, 2022.
For date range queries, we recommend using UTC seconds/milliseconds where possible (e.g. +created:>="2023-01-01T09:00:00+00:00").
Queries can use regular expressions as part of the query terms by wrapping the expression in forward slashes (/). To learn more about regular expressions, see the regexr and regex101 websites.
For example, to find matches with a game mode of either freeforall or capturetheflag the following query can be used: mode:/(freeforall|capturetheflag)/.
Using the operators detailed above you can craft queries with multiple terms to describe if a given result is returned or not returned. In addition, you can optionally define an ordering for the returned results by “boosting” the relative importance of the terms in your query.
This is done using ^ and an arbitrary booster number at the end of your query term(s).
For example, if using the query mode:freeforall^2 mode:capturetheflag to search for matches, your results will include both Free for All and Capture the Flag matches, but the Free for All results will be listed higher (i.e. be a “better” match) than the Capture the Flag results.
Note that since we didn’t use the + operator this query may also return results that are neither Free for All or Capture the Flag matches, but those results will be lowest of all.
The following character set must be escaped for accurate query results: +-=&|><!(){}[]^\"~*?:\\/ .
Note that the space character is included. For example, if you want to query for a “Capture The Flag” match use mode:Capture\ The\ Flag as opposed to mode:Capture The Flag. The latter query may still be valid but will produce the wrong results as it is searching for a match with a mode of “Capture”, and which contains “The” anywhere, and “Flag” anywhere.
If the match label contains an array players with the user IDs of all players currently in the match, we can use the following query to find any match a given friend is a part of:
Server
1
2
3
4
5
6
7
8
localnk=require("nakama")localquery="label.groups:<friend_user_id>"localmatches=nk.match_list(query)for_,matchinipairs(matches)donk.logger_info(string.format("Match id %s",match.match_id))end
Server
1
2
3
4
5
6
7
8
9
10
11
query:="+label.players:<friend_user_id>"matches,err:=nk.MatchList(ctx,query)iferr!=nil{logger.WithField("err",err).Error("Match listings error.")return}for_,match:=rangematches{logger.Info("Match id %s",match.MatchId)}
Server
1
2
3
4
5
6
7
8
functionfindFriendMatch(context: nkruntime.Context,logger: nkruntime.Logger,nk: nkruntime.Nakama){constquery="+label.players:<friend_user_id>";varmatches=nk.matchList(query);matches.forEach(function(match){logger.info("Match id '%s'",match.matchId);});}
If the match label contains an array groups with the IDs of all groups the user is part of, we can use the following query to find matches only from those groups:
Server
1
2
3
4
5
6
7
8
localnk=require("nakama")localquery="label.groups:/(<groupID>|<groupID2>|<groupID3)/"localmatches=nk.match_list(query)for_,matchinipairs(matches)donk.logger_info(string.format("Match id %s",match.match_id))end
Server
1
2
3
4
5
6
7
8
9
10
11
query:="label.groups:/(<groupID>|<groupID2>|<groupID3)/"matches,err:=nk.MatchList(ctx,query)iferr!=nil{logger.WithField("err",err).Error("Match listings error.")return}for_,match:=rangematches{logger.Info("Match id %s",match.MatchId)}
Server
1
2
3
4
5
6
7
8
functionfindGroupsMatch(context: nkruntime.Context,logger: nkruntime.Logger,nk: nkruntime.Nakama){constquery="label.groups:/(<groupID>|<groupID2>|<groupID3)/";varmatches=nk.matchList(query);matches.forEach(function(match){logger.info("Match id '%s'",match.matchId);});}
In this example, the match label contains an open value to indicate whether or not the match is accepting new players. This value would be updated in the match handler once the appropriate criteria has been met (e.g. enough players have joined or indicated that they are ready to begin).
Boolean Queries
Note than when querying boolean values, ’true’ is keyed to T and ‘false’ is keyed to F.
Server
1
2
3
4
5
6
7
8
localnk=require("nakama")localquery="+label.open:T label.game_mode:deathmatch"localmatches=nk.match_list(query)for_,matchinipairs(matches)donk.logger_info(string.format("Match id %s",match.match_id))end
Server
1
2
3
4
5
6
7
8
9
10
11
query:="+label.open:T label.game_mode:deathmatch"matches,err:=nk.MatchList(ctx,query)iferr!=nil{logger.WithField("err",err).Error("Match listings error.")return}for_,match:=rangematches{logger.Info("Match id %s",match.MatchId)}
Server
1
2
3
4
5
6
7
8
functionfindOpenMatches(context: nkruntime.Context,logger: nkruntime.Logger,nk: nkruntime.Nakama){constquery="+label.open:T label.game_mode:deathmatch";varmatches=nk.matchList(query);matches.forEach(function(match){logger.info("Match id '%s'",match.matchId);});}
In this example our player is located in the Europe region and prefers to matchmake with other players in the same region, followed by those in the Asia and Africa regions: