diff --git a/content/docs/howto/use-the-built-in-function-matchn/en.md b/content/docs/howto/use-the-built-in-function-matchn/en.md
new file mode 100644
index 0000000000..446fa58416
--- /dev/null
+++ b/content/docs/howto/use-the-built-in-function-matchn/en.md
@@ -0,0 +1,234 @@
+---
+title: Using the built-in function "matchN" as a field validator
+tags: [commented cue]
+authors: [jpluscplusm]
+toc_hide: true
+---
+
+{{{with _script_ "en" "HIDDEN: access to cue tip"}}}
+export PATH=/cues/$CUELANG_CUE_TIP:$PATH
+{{{end}}}
+
+{{}}
+This guide demonstrates a CUE feature that isn't yet available in a release.
+It is only available in the `cue` command and Go API when they are
+[installed from source]({{< relref "/docs/introduction/installation" >}}#install-from-source)
+using the `@master` version selector.
+{{}}
+
+This [Commented CUE]({{< relref "docs/howto#commented-cue-guides" >}})
+demonstrates how to use the
+[built-in]({{< relref "docs/reference/glossary#built-in-functions" >}})
+function `matchN()` as a field validator. This flexible function allows CUE
+constraints to be bound together in a wide range of combinations, including
+"one of", "any of", "all of", and "none of"/"not".
+
+This guide uses the following unreleased version of CUE:
+
+{{{with script "en" "cue version"}}}
+#ellipsis 1
+cue version
+{{{end}}}
+
+## Basic use
+
+The `matchN()` function takes two arguments:
+
+1. a **number constraint**;
+2. a **list of constraints**.
+
+The function validates a field's value by unifying the value with each item in
+the **list of constraints** in turn, and keeping count of how many list items
+the field's value is able to unify with. A field's value is valid if the count
+unifies successfully with the **number constraint**:
+
+
+{{{with upload "en" "basic"}}}
+-- basic.cue --
+package basic
+
+A: 42
+// A validates successfully.
+A: matchN(1, [int])
+A: matchN(2, [int, >10])
+A: matchN(2, [int, >10, >100])
+
+B: 42
+// B fails to validate.
+B: matchN(1, [int, >10])
+B: matchN(3, [int, >10, >100])
+{{{end}}}
+{{{with script "en" "basic"}}}
+! cue vet .:basic
+{{{end}}}
+
+## "One of"
+
+With its **number constraint** set to `1` the `matchN()` function checks that a
+field's value unifies successfully with just one of the **list of
+constraints**:
+
+{{{with upload "en" "one of"}}}
+-- one-of.cue --
+package oneOf
+
+A: 42
+// A validates successfully.
+A: matchN(1, [int])
+A: matchN(1, [>10, >100, string])
+
+B: 42
+// B fails to validate.
+B: matchN(1, [int, >10])
+B: matchN(1, [string, >100])
+{{{end}}}
+{{{with script "en" "one of"}}}
+! cue vet .:oneOf
+{{{end}}}
+
+## "Any of"
+
+If the **number constraint** is set to `>0`, the function checks that at least
+one of the **list of constraints** unifies with the field's value:
+
+{{{with upload "en" "any of"}}}
+-- any-of.cue --
+package anyOf
+
+A: 42
+// A validates successfully.
+A: matchN(>0, [>10])
+A: matchN(>0, [int, >0, >100, string])
+
+B: 42
+// B fails to validate.
+B: matchN(1, [int, >0])
+B: matchN(>0, [string, >100])
+{{{end}}}
+{{{with script "en" "any of"}}}
+! cue vet .:anyOf
+{{{end}}}
+
+## "All of"
+
+To check that a field's value unifies successfully with all of the **list of
+constraints**, set the **number constraint** to a value matching the number of
+items in the list:
+
+{{{with upload "en" "all of"}}}
+-- all-of.cue --
+package allOf
+
+import "math"
+
+A: 42
+// A validates successfully.
+A: matchN(1, [int])
+A: matchN(2, [int, >10])
+A: matchN(4, [int, >10, <100, math.MultipleOf(2)])
+
+B: 42
+// B fails to validate.
+B: matchN(3, [int, >10, >100])
+B: matchN(4, [int, >10, <100, math.MultipleOf(41)])
+{{{end}}}
+{{{with script "en" "all of"}}}
+! cue vet .:allOf
+{{{end}}}
+
+## "Not"
+
+If you set the **number constraint** to `0` then `matchN()` checks that a
+field's value doesn't unify successfully with any of the **list of
+constraints**:
+
+{{{with upload "en" "not"}}}
+-- not.cue --
+package not
+
+import (
+ "strings"
+ "struct"
+)
+
+A: 42
+// A validates successfully.
+A: matchN(0, [string])
+A: matchN(0, [bytes, struct.MinFields(0)])
+A: matchN(0, [>100, strings.HasPrefix("4")])
+
+B: 42
+// B fails to validate.
+B: matchN(0, [int])
+B: matchN(0, [string, number])
+B: matchN(0, [42, >100, strings.HasSuffix(2)])
+{{{end}}}
+{{{with script "en" "not"}}}
+! cue vet .:not
+{{{end}}}
+
+## More complex uses
+
+### References
+
+Either argument to `matchN()` can be resolved through a reference:
+
+{{{with upload "en" "all but one"}}}
+-- all-but-one.cue --
+package allButOne
+
+// A validates successfully.
+A: 42
+A: matchN(len(#C)-1, #C)
+
+// B fails to validate.
+B: 42.0
+B: matchN(len(#C)-1, #C)
+
+#C: [number, int, >100]
+{{{end}}}
+{{{with script "en" "all but one"}}}
+! cue vet .:allButOne
+{{{end}}}
+
+### Composite data structures
+
+The `matchN()` function can validate composite data structures, not just
+primitive values. Use it with both structs and lists:
+
+{{{with upload "en" "composite"}}}
+-- composite.cue --
+package composite
+
+// A validates successfully.
+A: matchN(>0, [#C1, #C2]) & {
+ x: 42
+ y: "42"
+}
+// B fails to validate.
+B: matchN(>0, [#C1, #C2]) & {
+ x: "4.2"
+ y: "4.2"
+ z: "4.2"
+}
+#C1: {x: int, ...}
+#C2: {
+ z: float
+ y: string
+ ...
+}
+
+// D validates successfully.
+D: [1, 2, 3] & matchN(1, [#F1, #F2, #F3])
+// E fails to validate.
+E: [11, 12, 13] & matchN(1, [#F1, #F2, #F3])
+#F1: [...>0]
+#F2: [...>10]
+#F3: [...>100]
+{{{end}}}
+{{{with script "en" "composite"}}}
+! cue vet .:composite
+{{{end}}}
+
+The sub-optimal error reporting for field `E` is tracked in {{}}.
diff --git a/content/docs/howto/use-the-built-in-function-matchn/gen_cache.cue b/content/docs/howto/use-the-built-in-function-matchn/gen_cache.cue
new file mode 100644
index 0000000000..3a79e87330
--- /dev/null
+++ b/content/docs/howto/use-the-built-in-function-matchn/gen_cache.cue
@@ -0,0 +1,156 @@
+package site
+{
+ content: {
+ docs: {
+ howto: {
+ "use-the-built-in-function-matchn": {
+ page: {
+ cache: {
+ upload: {
+ basic: "pgKy7ZjgCtqS8DgsHmRsK2nBYeSjoOcdRjq7UmRXwOQ="
+ "one of": "yDaX7Fb0+gk3jQdRfEEbEk02wW1H+hg0Z1jj1HYPBVE="
+ "any of": "165Tc+q1AeA6opEPagvsLc5/IamRMiypLlHULC0jPw0="
+ "all of": "pEr9Rh3V5+oYsvqeSR9yl5N9Rw+YSgc4lZ2Q6WAh+j0="
+ not: "IECXMjtkVDxqyNERUWzBzE5lKeYCVMhOyeeZhvNsEgI="
+ "all but one": "XseOBFuulZdMc7zkzoiySQex9kJs3Y5D2GYncddt7c8="
+ composite: "IrMJLK5IergqI2YDo/S1xmQUqsIrM7ExisoQAbHPxW0="
+ }
+ multi_step: {
+ hash: "4EDOB4V3CVCU8ML21G75L9MFRJCPA7R35ENF28J833TLGF2RNGKG===="
+ scriptHash: "DL630JFRN3O5DC698D9VGH8I0JD9E68M3E94FN3M3TU14GCR96K0===="
+ steps: [{
+ doc: ""
+ cmd: "export PATH=/cues/v0.11.0-0.dev.0.20240820111527-0a41a20985d3:$PATH"
+ exitCode: 0
+ output: ""
+ }, {
+ doc: "#ellipsis 1"
+ cmd: "cue version"
+ exitCode: 0
+ output: """
+ cue version v0.11.0-0.dev.0.20240820111527-0a41a20985d3
+ ...
+
+ """
+ }, {
+ doc: ""
+ cmd: "cue vet .:basic"
+ exitCode: 1
+ output: """
+ B: invalid value 42 (does not satisfy matchN(1, [int,>10])): 2 matched, expected 1:
+ ./basic.cue:11:4
+ ./basic.cue:9:4
+ ./basic.cue:11:11
+ ./basic.cue:12:4
+ B: invalid value 42 (does not satisfy matchN(3, [int,>10,>100])): 2 matched, expected 3:
+ ./basic.cue:12:4
+ ./basic.cue:9:4
+ ./basic.cue:11:4
+ ./basic.cue:12:11
+
+ """
+ }, {
+ doc: ""
+ cmd: "cue vet .:oneOf"
+ exitCode: 1
+ output: """
+ B: invalid value 42 (does not satisfy matchN(1, [string,>100])): 0 matched, expected 1:
+ ./one-of.cue:11:4
+ ./one-of.cue:8:4
+ ./one-of.cue:10:4
+ ./one-of.cue:11:11
+
+ """
+ }, {
+ doc: ""
+ cmd: "cue vet .:anyOf"
+ exitCode: 1
+ output: """
+ B: invalid value 42 (does not satisfy matchN(1, [int,>0])): 2 matched, expected 1:
+ ./any-of.cue:10:4
+ ./any-of.cue:8:4
+ ./any-of.cue:10:11
+ ./any-of.cue:11:4
+ B: invalid value 42 (does not satisfy matchN(>0, [string,>100])): 0 matched, expected >0:
+ ./any-of.cue:11:4
+ ./any-of.cue:8:4
+ ./any-of.cue:10:4
+ ./any-of.cue:11:11
+
+ """
+ }, {
+ doc: ""
+ cmd: "cue vet .:allOf"
+ exitCode: 1
+ output: """
+ B: invalid value 42 (does not satisfy matchN(3, [int,>10,>100])): 2 matched, expected 3:
+ ./all-of.cue:13:4
+ ./all-of.cue:11:4
+ ./all-of.cue:13:11
+ ./all-of.cue:14:4
+ B: invalid value 42 (does not satisfy matchN(4, [int,>10,<100,math.MultipleOf(41)])): 3 matched, expected 4:
+ ./all-of.cue:14:4
+ ./all-of.cue:11:4
+ ./all-of.cue:13:4
+ ./all-of.cue:14:11
+
+ """
+ }, {
+ doc: ""
+ cmd: "cue vet .:not"
+ exitCode: 1
+ output: """
+ B: invalid value 42 (does not satisfy matchN(0, [int])): 1 matched, expected 0:
+ ./not.cue:16:4
+ ./not.cue:14:4
+ ./not.cue:16:11
+ ./not.cue:17:4
+ ./not.cue:18:4
+ B: invalid value 42 (does not satisfy matchN(0, [string,number])): 1 matched, expected 0:
+ ./not.cue:17:4
+ ./not.cue:14:4
+ ./not.cue:16:4
+ ./not.cue:17:11
+ ./not.cue:18:4
+ B: invalid value 42 (does not satisfy matchN(0, [42,>100,strings.HasSuffix(2)])): 1 matched, expected 0:
+ ./not.cue:18:4
+ ./not.cue:14:4
+ ./not.cue:16:4
+ ./not.cue:17:4
+ ./not.cue:18:11
+
+ """
+ }, {
+ doc: ""
+ cmd: "cue vet .:allButOne"
+ exitCode: 1
+ output: """
+ B: invalid value 42.0 (does not satisfy matchN(2, [number,int,>100])): 1 matched, expected 2:
+ ./all-but-one.cue:9:4
+ ./all-but-one.cue:8:4
+ ./all-but-one.cue:9:11
+
+ """
+ }, {
+ doc: ""
+ cmd: "cue vet .:composite"
+ exitCode: 1
+ output: """
+ B: invalid value {x:"4.2",y:"4.2",z:"4.2"} (does not satisfy matchN(>0, [{x:int},{z:float,y:string}])): 0 matched, expected >0:
+ ./composite.cue:9:4
+ ./composite.cue:9:11
+ E: invalid value [11,12,13] (does not satisfy matchN(1, [[],[],[]])): 2 matched, expected 1:
+ ./composite.cue:24:19
+ ./composite.cue:24:4
+ ./composite.cue:24:26
+
+ """
+ }]
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+}
diff --git a/content/docs/howto/use-the-built-in-function-matchn/page.cue b/content/docs/howto/use-the-built-in-function-matchn/page.cue
new file mode 100644
index 0000000000..d1b59cb948
--- /dev/null
+++ b/content/docs/howto/use-the-built-in-function-matchn/page.cue
@@ -0,0 +1,3 @@
+package site
+
+content: docs: howto: "use-the-built-in-function-matchn": page: _
diff --git a/content/docs/introduction/installation/en.md b/content/docs/introduction/installation/en.md
index 83f2e782be..2528ebb4c3 100644
--- a/content/docs/introduction/installation/en.md
+++ b/content/docs/introduction/installation/en.md
@@ -31,7 +31,7 @@ that's appropriate for your operating system.
On Linux, Microsoft Windows, and macOS, the `cue` command can be downloaded from the
[official CUE releases](https://github.com/cue-lang/cue/releases/).
-These releases include *prereleases*, which are cutting-edge versions of `cue`
+These releases include *pre-releases*, which are cutting-edge versions of `cue`
made available to help expose bugs and flush out unintended behaviours.
#### Install using Homebrew
@@ -64,16 +64,32 @@ docker pull cuelang/cue:latest
On
[platforms supported by Go](https://go.dev/dl/#stable),
-`cue` can be installed from source using any of its released versions.
+`cue` can be installed from source using any of its
+release, pre-release, or as-yet-unreleased versions.
+Installing from source requires that you already have
+[Go](https://go.dev)
+installed and available.
-For example, to fetch the latest version:
+For example, to fetch the `latest` version:
-{{{with script "en" "go install"}}}
+{{{with script "en" "go install cmd/cue@latest"}}}
#ellipsis 0
go install cuelang.org/go/cmd/cue@latest
{{{end}}}
-This method requires that you already have [Go](https://go.dev) installed.
+[This page](https://pkg.go.dev/cuelang.org/go?tab=versions)
+lists the installable releases and pre-releases that you can specify instead of
+`latest`.
+
+You can also install the tip version of `cue` by specifying `master`:
+
+{{{with script "en" "go install cmd/cue@master"}}}
+#ellipsis 0
+go install cuelang.org/go/cmd/cue@master
+{{{end}}}
+
+The capabilities of the tip version change frequently because it contains the
+most recent, unreleased code.
#### Install on Arch Linux
diff --git a/content/docs/introduction/installation/gen_cache.cue b/content/docs/introduction/installation/gen_cache.cue
index 15a09e5061..fc160a190e 100644
--- a/content/docs/introduction/installation/gen_cache.cue
+++ b/content/docs/introduction/installation/gen_cache.cue
@@ -7,8 +7,8 @@ package site
page: {
cache: {
multi_step: {
- hash: "7URNEHKR89PNIBH2CGQN4B1CHAUP2EG44GBURO11CDB9LEJ7R3AG===="
- scriptHash: "D37M5I4CC9UQ54DHIBSQ5TTE5SQT8FQRIFIR5UV0EUS1PEIFG7QG===="
+ hash: "9KNFG0L44SGMJ4BSGBOD55AU4467B42KN35H3D1G6TRQAMUCIU0G===="
+ scriptHash: "T22RVEVNCSL1EBGJC1VB4CUC6NORBUOCAN65A6USMPUL369LHPLG===="
steps: [{
doc: ""
cmd: "export GOMODCACHE=/caches/gomodcache"
@@ -28,6 +28,14 @@ package site
"""
}, {
+ doc: "#ellipsis 0"
+ cmd: "go install cuelang.org/go/cmd/cue@master"
+ exitCode: 0
+ output: """
+ ...
+
+ """
+ }, {
doc: ""
cmd: "go mod init go.example"
exitCode: 0
diff --git a/hugo/content/en/docs/howto/use-the-built-in-function-matchn/index.md b/hugo/content/en/docs/howto/use-the-built-in-function-matchn/index.md
new file mode 100644
index 0000000000..17eb72991a
--- /dev/null
+++ b/hugo/content/en/docs/howto/use-the-built-in-function-matchn/index.md
@@ -0,0 +1,295 @@
+---
+title: Using the built-in function "matchN" as a field validator
+tags: [commented cue]
+authors: [jpluscplusm]
+toc_hide: true
+---
+
+{{}}
+This guide demonstrates a CUE feature that isn't yet available in a release.
+It is only available in the `cue` command and Go API when they are
+[installed from source]({{< relref "/docs/introduction/installation" >}}#install-from-source)
+using the `@master` version selector.
+{{}}
+
+This [Commented CUE]({{< relref "docs/howto#commented-cue-guides" >}})
+demonstrates how to use the
+[built-in]({{< relref "docs/reference/glossary#built-in-functions" >}})
+function `matchN()` as a field validator. This flexible function allows CUE
+constraints to be bound together in a wide range of combinations, including
+"one of", "any of", "all of", and "none of"/"not".
+
+This guide uses the following unreleased version of CUE:
+
+```text { title="TERMINAL" type="terminal" codeToCopy="Y3VlIHZlcnNpb24=" }
+$ cue version
+cue version v0.11.0-0.dev.0.20240820111527-0a41a20985d3
+...
+```
+
+## Basic use
+
+The `matchN()` function takes two arguments:
+
+1. a **number constraint**;
+2. a **list of constraints**.
+
+The function validates a field's value by unifying the value with each item in
+the **list of constraints** in turn, and keeping count of how many list items
+the field's value is able to unify with. A field's value is valid if the count
+unifies successfully with the **number constraint**:
+
+
+{{< code-tabs >}}
+{{< code-tab name="basic.cue" language="cue" area="top-left" >}}
+package basic
+
+A: 42
+// A validates successfully.
+A: matchN(1, [int])
+A: matchN(2, [int, >10])
+A: matchN(2, [int, >10, >100])
+
+B: 42
+// B fails to validate.
+B: matchN(1, [int, >10])
+B: matchN(3, [int, >10, >100])
+{{< /code-tab >}}{{< /code-tabs >}}
+```text { title="TERMINAL" type="terminal" codeToCopy="Y3VlIHZldCAuOmJhc2lj" }
+$ cue vet .:basic
+B: invalid value 42 (does not satisfy matchN(1, [int,>10])): 2 matched, expected 1:
+ ./basic.cue:11:4
+ ./basic.cue:9:4
+ ./basic.cue:11:11
+ ./basic.cue:12:4
+B: invalid value 42 (does not satisfy matchN(3, [int,>10,>100])): 2 matched, expected 3:
+ ./basic.cue:12:4
+ ./basic.cue:9:4
+ ./basic.cue:11:4
+ ./basic.cue:12:11
+```
+
+## "One of"
+
+With its **number constraint** set to `1` the `matchN()` function checks that a
+field's value unifies successfully with just one of the **list of
+constraints**:
+
+{{< code-tabs >}}
+{{< code-tab name="one-of.cue" language="cue" area="top-left" >}}
+package oneOf
+
+A: 42
+// A validates successfully.
+A: matchN(1, [int])
+A: matchN(1, [>10, >100, string])
+
+B: 42
+// B fails to validate.
+B: matchN(1, [int, >10])
+B: matchN(1, [string, >100])
+{{< /code-tab >}}{{< /code-tabs >}}
+```text { title="TERMINAL" type="terminal" codeToCopy="Y3VlIHZldCAuOm9uZU9m" }
+$ cue vet .:oneOf
+B: invalid value 42 (does not satisfy matchN(1, [string,>100])): 0 matched, expected 1:
+ ./one-of.cue:11:4
+ ./one-of.cue:8:4
+ ./one-of.cue:10:4
+ ./one-of.cue:11:11
+```
+
+## "Any of"
+
+If the **number constraint** is set to `>0`, the function checks that at least
+one of the **list of constraints** unifies with the field's value:
+
+{{< code-tabs >}}
+{{< code-tab name="any-of.cue" language="cue" area="top-left" >}}
+package anyOf
+
+A: 42
+// A validates successfully.
+A: matchN(>0, [>10])
+A: matchN(>0, [int, >0, >100, string])
+
+B: 42
+// B fails to validate.
+B: matchN(1, [int, >0])
+B: matchN(>0, [string, >100])
+{{< /code-tab >}}{{< /code-tabs >}}
+```text { title="TERMINAL" type="terminal" codeToCopy="Y3VlIHZldCAuOmFueU9m" }
+$ cue vet .:anyOf
+B: invalid value 42 (does not satisfy matchN(1, [int,>0])): 2 matched, expected 1:
+ ./any-of.cue:10:4
+ ./any-of.cue:8:4
+ ./any-of.cue:10:11
+ ./any-of.cue:11:4
+B: invalid value 42 (does not satisfy matchN(>0, [string,>100])): 0 matched, expected >0:
+ ./any-of.cue:11:4
+ ./any-of.cue:8:4
+ ./any-of.cue:10:4
+ ./any-of.cue:11:11
+```
+
+## "All of"
+
+To check that a field's value unifies successfully with all of the **list of
+constraints**, set the **number constraint** to a value matching the number of
+items in the list:
+
+{{< code-tabs >}}
+{{< code-tab name="all-of.cue" language="cue" area="top-left" >}}
+package allOf
+
+import "math"
+
+A: 42
+// A validates successfully.
+A: matchN(1, [int])
+A: matchN(2, [int, >10])
+A: matchN(4, [int, >10, <100, math.MultipleOf(2)])
+
+B: 42
+// B fails to validate.
+B: matchN(3, [int, >10, >100])
+B: matchN(4, [int, >10, <100, math.MultipleOf(41)])
+{{< /code-tab >}}{{< /code-tabs >}}
+```text { title="TERMINAL" type="terminal" codeToCopy="Y3VlIHZldCAuOmFsbE9m" }
+$ cue vet .:allOf
+B: invalid value 42 (does not satisfy matchN(3, [int,>10,>100])): 2 matched, expected 3:
+ ./all-of.cue:13:4
+ ./all-of.cue:11:4
+ ./all-of.cue:13:11
+ ./all-of.cue:14:4
+B: invalid value 42 (does not satisfy matchN(4, [int,>10,<100,math.MultipleOf(41)])): 3 matched, expected 4:
+ ./all-of.cue:14:4
+ ./all-of.cue:11:4
+ ./all-of.cue:13:4
+ ./all-of.cue:14:11
+```
+
+## "Not"
+
+If you set the **number constraint** to `0` then `matchN()` checks that a
+field's value doesn't unify successfully with any of the **list of
+constraints**:
+
+{{< code-tabs >}}
+{{< code-tab name="not.cue" language="cue" area="top-left" >}}
+package not
+
+import (
+ "strings"
+ "struct"
+)
+
+A: 42
+// A validates successfully.
+A: matchN(0, [string])
+A: matchN(0, [bytes, struct.MinFields(0)])
+A: matchN(0, [>100, strings.HasPrefix("4")])
+
+B: 42
+// B fails to validate.
+B: matchN(0, [int])
+B: matchN(0, [string, number])
+B: matchN(0, [42, >100, strings.HasSuffix(2)])
+{{< /code-tab >}}{{< /code-tabs >}}
+```text { title="TERMINAL" type="terminal" codeToCopy="Y3VlIHZldCAuOm5vdA==" }
+$ cue vet .:not
+B: invalid value 42 (does not satisfy matchN(0, [int])): 1 matched, expected 0:
+ ./not.cue:16:4
+ ./not.cue:14:4
+ ./not.cue:16:11
+ ./not.cue:17:4
+ ./not.cue:18:4
+B: invalid value 42 (does not satisfy matchN(0, [string,number])): 1 matched, expected 0:
+ ./not.cue:17:4
+ ./not.cue:14:4
+ ./not.cue:16:4
+ ./not.cue:17:11
+ ./not.cue:18:4
+B: invalid value 42 (does not satisfy matchN(0, [42,>100,strings.HasSuffix(2)])): 1 matched, expected 0:
+ ./not.cue:18:4
+ ./not.cue:14:4
+ ./not.cue:16:4
+ ./not.cue:17:4
+ ./not.cue:18:11
+```
+
+## More complex uses
+
+### References
+
+Either argument to `matchN()` can be resolved through a reference:
+
+{{< code-tabs >}}
+{{< code-tab name="all-but-one.cue" language="cue" area="top-left" >}}
+package allButOne
+
+// A validates successfully.
+A: 42
+A: matchN(len(#C)-1, #C)
+
+// B fails to validate.
+B: 42.0
+B: matchN(len(#C)-1, #C)
+
+#C: [number, int, >100]
+{{< /code-tab >}}{{< /code-tabs >}}
+```text { title="TERMINAL" type="terminal" codeToCopy="Y3VlIHZldCAuOmFsbEJ1dE9uZQ==" }
+$ cue vet .:allButOne
+B: invalid value 42.0 (does not satisfy matchN(2, [number,int,>100])): 1 matched, expected 2:
+ ./all-but-one.cue:9:4
+ ./all-but-one.cue:8:4
+ ./all-but-one.cue:9:11
+```
+
+### Composite data structures
+
+The `matchN()` function can validate composite data structures, not just
+primitive values. Use it with both structs and lists:
+
+{{< code-tabs >}}
+{{< code-tab name="composite.cue" language="cue" area="top-left" >}}
+package composite
+
+// A validates successfully.
+A: matchN(>0, [#C1, #C2]) & {
+ x: 42
+ y: "42"
+}
+// B fails to validate.
+B: matchN(>0, [#C1, #C2]) & {
+ x: "4.2"
+ y: "4.2"
+ z: "4.2"
+}
+#C1: {x: int, ...}
+#C2: {
+ z: float
+ y: string
+ ...
+}
+
+// D validates successfully.
+D: [1, 2, 3] & matchN(1, [#F1, #F2, #F3])
+// E fails to validate.
+E: [11, 12, 13] & matchN(1, [#F1, #F2, #F3])
+#F1: [...>0]
+#F2: [...>10]
+#F3: [...>100]
+{{< /code-tab >}}{{< /code-tabs >}}
+```text { title="TERMINAL" type="terminal" codeToCopy="Y3VlIHZldCAuOmNvbXBvc2l0ZQ==" }
+$ cue vet .:composite
+B: invalid value {x:"4.2",y:"4.2",z:"4.2"} (does not satisfy matchN(>0, [{x:int},{z:float,y:string}])): 0 matched, expected >0:
+ ./composite.cue:9:4
+ ./composite.cue:9:11
+E: invalid value [11,12,13] (does not satisfy matchN(1, [[],[],[]])): 2 matched, expected 1:
+ ./composite.cue:24:19
+ ./composite.cue:24:4
+ ./composite.cue:24:26
+```
+
+The sub-optimal error reporting for field `E` is tracked in {{}}.
diff --git a/hugo/content/en/docs/introduction/installation/index.md b/hugo/content/en/docs/introduction/installation/index.md
index a14346548e..7489ea3d19 100644
--- a/hugo/content/en/docs/introduction/installation/index.md
+++ b/hugo/content/en/docs/introduction/installation/index.md
@@ -26,7 +26,7 @@ that's appropriate for your operating system.
On Linux, Microsoft Windows, and macOS, the `cue` command can be downloaded from the
[official CUE releases](https://github.com/cue-lang/cue/releases/).
-These releases include *prereleases*, which are cutting-edge versions of `cue`
+These releases include *pre-releases*, which are cutting-edge versions of `cue`
made available to help expose bugs and flush out unintended behaviours.
#### Install using Homebrew
@@ -57,16 +57,32 @@ $ docker pull cuelang/cue:latest
On
[platforms supported by Go](https://go.dev/dl/#stable),
-`cue` can be installed from source using any of its released versions.
+`cue` can be installed from source using any of its
+release, pre-release, or as-yet-unreleased versions.
+Installing from source requires that you already have
+[Go](https://go.dev)
+installed and available.
-For example, to fetch the latest version:
+For example, to fetch the `latest` version:
```text { title="TERMINAL" type="terminal" codeToCopy="Z28gaW5zdGFsbCBjdWVsYW5nLm9yZy9nby9jbWQvY3VlQGxhdGVzdA==" }
$ go install cuelang.org/go/cmd/cue@latest
...
```
-This method requires that you already have [Go](https://go.dev) installed.
+[This page](https://pkg.go.dev/cuelang.org/go?tab=versions)
+lists the installable releases and pre-releases that you can specify instead of
+`latest`.
+
+You can also install the tip version of `cue` by specifying `master`:
+
+```text { title="TERMINAL" type="terminal" codeToCopy="Z28gaW5zdGFsbCBjdWVsYW5nLm9yZy9nby9jbWQvY3VlQG1hc3Rlcg==" }
+$ go install cuelang.org/go/cmd/cue@master
+...
+```
+
+The capabilities of the tip version change frequently because it contains the
+most recent, unreleased code.
#### Install on Arch Linux