diff --git a/demo/resource-usage.demo b/demo/resource-usage.demo
new file mode 100644
index 000000000..5aa96215a
--- /dev/null
+++ b/demo/resource-usage.demo
@@ -0,0 +1,28 @@
+# Let's getting started with kwokctl!
+kwokctl create cluster --enable-metrics-server -c ./kustomize/metrics/resource/metrics-resource.yaml -c ./kustomize/metrics/usage/usage-from-annotation.yaml
+
+kwokctl scale node --replicas 2
+kwokctl scale pod --replicas 8
+
+# Wait for a while to let the metrics server collect the metrics.
+sleep 45
+
+# Now we can check the metrics.
+kubectl top node
+kubectl top pod
+
+# Let's add some usage metrics to the pods.
+kubectl patch pod pod-000000 --type=json -p='[{"op":"add","path":"/metadata/annotations","value":{"kwok.x-k8s.io/usage-cpu":"10000m","kwok.x-k8s.io/usage-memory":"10000Mi"}}]'
+
+# Wait for a while to let the metrics server collect the metrics.
+sleep 15
+
+# Now we can check the metrics again.
+kubectl top node
+kubectl top pod
+
+# Delete the cluster.
+kwokctl delete cluster
+
+# That's all, enjoy it!
+clear
diff --git a/demo/resource-usage.svg b/demo/resource-usage.svg
new file mode 100644
index 000000000..319baeec2
--- /dev/null
+++ b/demo/resource-usage.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/kustomize/metrics/resource/README.md b/kustomize/metrics/resource/README.md
new file mode 100644
index 000000000..b0201d59c
--- /dev/null
+++ b/kustomize/metrics/resource/README.md
@@ -0,0 +1,5 @@
+# Metrics Resource
+
+This Metrics simulates kubelet's `/metrics/resource` endpoint.
+Please refer to [Metrics](https://kwok.sigs.k8s.io/docs/user/metrics-configuration) for more on how it works.
+
diff --git a/kustomize/metrics/usage/README.md b/kustomize/metrics/usage/README.md
new file mode 100644
index 000000000..23d2d24d6
--- /dev/null
+++ b/kustomize/metrics/usage/README.md
@@ -0,0 +1,8 @@
+# Resource Usage
+
+This ResourceUsage simulates the resource usage of Pod(s) based on information collected from the respective annotations.
+
+Provided two annotations for Pod(s):
+
+- `kwok.x-k8s.io/usage-cpu`
+- `kwok.x-k8s.io/usage-memory`
diff --git a/site/config.yaml b/site/config.yaml
index 8f105bc2f..873003345 100644
--- a/site/config.yaml
+++ b/site/config.yaml
@@ -215,6 +215,18 @@ menu:
- identifier: attach
pageRef: "/docs/user/attach-configuration"
parent: configuration
+ - identifier: metrics
+ pageRef: "/docs/user/metrics-configuration"
+ parent: configuration
+ - identifier: resource-usage
+ pageRef: "/docs/user/resource-usage-configuration"
+ parent: configuration
+ - identifier: go-template
+ pageRef: "/docs/user/go-template"
+ parent: configuration
+ - identifier: cel-expressions
+ pageRef: "/docs/user/cel-expressions"
+ parent: configuration
# Design Children
- identifier: introduction
diff --git a/site/content/en/docs/user/_index.md b/site/content/en/docs/user/_index.md
index 46814c02e..0d38a7f70 100644
--- a/site/content/en/docs/user/_index.md
+++ b/site/content/en/docs/user/_index.md
@@ -37,6 +37,7 @@ If any special concerns, you can configure KWOK with options and stages.
- [Exec]
- [Logs]
- [Attach]
+- [ResourceUsage]
I hope this helps you get started with KWOK! Good luck and have fun!
@@ -53,3 +54,4 @@ I hope this helps you get started with KWOK! Good luck and have fun!
[Exec]: {{< relref "/docs/user/exec-configuration" >}}
[Logs]: {{< relref "/docs/user/logs-configuration" >}}
[Attach]: {{< relref "/docs/user/attach-configuration" >}}
+[ResourceUsage]: {{< relref "/docs/user/resource-usage-configuration" >}}
diff --git a/site/content/en/docs/user/cel-expressions.md b/site/content/en/docs/user/cel-expressions.md
new file mode 100644
index 000000000..f8133c498
--- /dev/null
+++ b/site/content/en/docs/user/cel-expressions.md
@@ -0,0 +1,77 @@
+---
+title: "CEL Expressions in `kwok`"
+---
+
+# Notes on CEL Expressions in `kwok`
+
+The page provides a concise note on writing CEL expressions in `kwok` CRs.
+
+Below is the list of all CRs in `kwok` that contains CEL based fields.
+* [Metric]
+* [ResourceUsage]
+* [ClusterResourceUsage]
+
+
+You must follow [the CEL language specification] when writing the expressions.
+For predefined functions of CEL, please refer to [CEL predefined functions].
+
+Besides the built-in functions, `kwok` also provides some customized extension functions.
+An exhaustive list of all the extension functions with their usages is given below.
+
+* `Now()`: takes no parameters and returns the current timestamp.
+* `Rand()`: takes no parameters and returns a random `float64` value.
+* `SinceSecond()` returns the seconds elapsed since a given resource (`pod` or `node`) was created.
+ For example: `SinceSecond(pod)`, `node.SinceSecond(node)`.
+* `UnixSecond()` returns the Unix time of a given time of type `time.Time`.
+ For example: , `UnixSecond(Now())`, `UnixSecond(node.metadata.creationTimestamp)`.
+* `Quantity()` returns a float64 value of a given Quantity value. For example: `Quantity("100m")`, `Quantity("10Mi")`.
+* `Usage()` returns the current instantaneous resource usage with the simulation data in [ResourceUsage (ClusterResourceUsage)].
+ For example: `Usage(pod, "memory")`, `Usage(node, "memory")`, `Usage(pod, "memory", container.name)` return the
+ current working set of a resource (pod, node or container) in bytes.
+* `CumulativeUsage()` returns the cumulative resource usage in seconds with the simulation data given in [ResourceUsage (ClusterResourceUsage)].
+ For example: `CumulativeUsage(pod, "cpu")`, `CumulativeUsage(node, "cpu")`, `CumulativeUsage(pod, "cpu", container.name)`
+ return a cumulative cpu time consumed by a resource (pod, node or container) in core-seconds.
+
+Additionally, `kwok` provides three special CEL variables `node`, `pod`, and `container` that could be used
+in the expressions.
+The three variables are set to the corresponding node, pod, container resource object respectively and users can
+reference any nested fields of the resource objects simply via the CEL field selection expression (`e.f` format).
+For example, you could use expression `node.metadata.name` to obtain the node name.
+
+{{< hint "info" >}}
+
+The functions with at least one parameter can be called in a receiver call-style.
+That is, a function call like `f(e1, e2)` can also be called in style `e1.f(e2)`. For example, you can use `pod.Usage("memory")`
+as an alternative to `Usage(pod, "memory")`.
+
+{{< /hint >}}
+
+
+It is worth noting that the use of some extension functions is restricted to specific CRs and contexts in the sense
+that they are not generic but designed for special evaluating tasks.
+The detailed limitations are described below.
+
+## Functions Limitation
+
+Function `Usage()` and `CumulativeUsage()` can only be used in the Metric resource.
+For other functions listed above, users are also allowed to use them in ResourceUsage and ClusterResourceUsage
+to build dynamic resource usage patterns.
+
+The reason behind is that when `kwok` evaluates functions `Usage()` or `CumulativeUsage()`,
+it actually takes the simulation data given in ResourceUsage and ClusterResourceUsage to obtain metric values.
+Therefore, please ensure that the associated ResourceUsage or ClusterResourceUsage with the needed resource types
+(cpu or memory) are also provided when using function `Usage()` and `CumulativeUsage()`.
+
+## Variables Limitation
+
+When using the three special CEL variables `node`, `pod`, and `container` in Metric resource, you should follow the below rules.
+* When `dimension` is `node`: only `node` variable can be used.
+* When `dimension` is `pod`: only `node`, `pod` can be used.
+* When `dimension` is `container`: `node`, `pod`, `container` all can be used.
+
+
+[Metric]: {{< relref "/docs/generated/apis" >}}#kwok.x-k8s.io/v1alpha1.Metric
+[ResourceUsage]: {{< relref "/docs/generated/apis" >}}#kwok.x-k8s.io/v1alpha1.ResourceUsage
+[ClusterResourceUsage]: {{< relref "/docs/generated/apis" >}}#kwok.x-k8s.io/v1alpha1.ClusterResourceUsage
+[the CEL language specification]: https://github.com/google/cel-spec/blob/master/doc/langdef.md
+[CEL predefined functions]: https://github.com/google/cel-spec/blob/master/doc/langdef.md#list-of-standard-definitions
diff --git a/site/content/en/docs/user/go-template.md b/site/content/en/docs/user/go-template.md
new file mode 100644
index 000000000..51b6a27ba
--- /dev/null
+++ b/site/content/en/docs/user/go-template.md
@@ -0,0 +1,26 @@
+---
+title: "Go Template in `kwok`"
+---
+
+# Notes on Go Template in `kwok`
+
+
+The page provides a concise note on writing go templates in kwok CRs.
+
+
+Currently, only `Stage` CR has go template based fields (`Spec.Next.StatusTemplate`).
+
+
+You must follow [the go text template syntax] when writing the templates.
+For predefined functions of go text template, please refer to [go text template functions].
+Besides the built-in functions, `kwok` also supports [sprig template functions].
+
+It is worth noting that the "context" (which is denoted by the period character `.` ) to a template in `kwok` is set to the
+referenced Kubernetes resource.
+For example, you can use `.metadata.name` in a template to obtain the corresponding Kubernetes resource name.
+
+
+
+[the go text template syntax]: https://pkg.go.dev/text/template
+[go text template functions]: https://pkg.go.dev/text/template#hdr-Functions
+[sprig template functions]: https://masterminds.github.io/sprig/
diff --git a/site/content/en/docs/user/kwok-in-cluster.md b/site/content/en/docs/user/kwok-in-cluster.md
index 5909ed287..7a2e38bfa 100644
--- a/site/content/en/docs/user/kwok-in-cluster.md
+++ b/site/content/en/docs/user/kwok-in-cluster.md
@@ -35,6 +35,21 @@ NOTE: This configures the pod/node emulation behavior, if not it will do nothing
kubectl apply -f "https://github.com/${KWOK_REPO}/releases/download/${KWOK_LATEST_RELEASE}/stage-fast.yaml"
```
+## Set up default CRs of resource usage (optional)
+
+This allows to simulate the resource usage of nodes, pods and containers.
+
+``` bash
+kubectl apply -f "https://github.com/${KWOK_REPO}/releases/download/${KWOK_LATEST_RELEASE}/metrics-usage.yaml"
+```
+
+The above configuration sets the CPU and memory usage of all the containers managed by `kwok` to `1m` and to `1Mi` respectively.
+To override the defaults, you can add annotation `"kwok.x-k8s.io/usage-cpu"` (for cpu usage) and
+`"kwok.x-k8s.io/usage-memory"` (for memory usage) with any quantity value you want to the fake pods.
+
+The resource usage simulation used above is annotation-based and the configuration is available at [here][resource usage from annotation].
+For the explanation of how it works and more complex resource usage simulation methods, please refer to [ResourceUsage configuration].
+
## Old way to deploy kwok
Old way to deploy kwok is [here][kwok in cluster old].
@@ -45,3 +60,5 @@ Now, you can use `kwok` to [manage nodes and pods] in the Kubernetes cluster.
[manage nodes and pods]: {{< relref "/docs/user/kwok-manage-nodes-and-pods" >}}
[kwok in cluster old]: {{< relref "/docs/user/kwok-in-cluster-old" >}}
+[resource usage from annotation]: https://github.com/kubernetes-sigs/kwok/tree/main/kustomize/metrics/usage
+[ResourceUsage configuration]: {{< relref "/docs/user/resource-usage-configuration" >}}
diff --git a/site/content/en/docs/user/metrics-configuration.md b/site/content/en/docs/user/metrics-configuration.md
new file mode 100644
index 000000000..de70665f2
--- /dev/null
+++ b/site/content/en/docs/user/metrics-configuration.md
@@ -0,0 +1,108 @@
+---
+title: "Metrics"
+---
+
+# Metrics Configuration
+
+{{< hint "info" >}}
+
+This document walks you through how to configure the Metrics feature.
+
+{{< /hint >}}
+
+## What is a Metrics?
+
+The [Metrics] is a [`kwok` Configuration][configuration] that allows users to define and simulate metrics endpoints exposed by kubelet.
+
+The YAML below shows all the fields of a Metrics resource:
+
+``` yaml
+kind: Metrics
+apiVersion: kwok.x-k8s.io/v1alpha1
+metadata:
+ name:
+spec:
+ path:
+ metrics:
+ - name:
+ help:
+ kind:
+ dimension:
+ labels:
+ - name:
+ value:
+ value: # for counter and gauge
+ buckets: # for histogram
+ - le:
+ value:
+ hidden:
+```
+
+There are total four metric-related endpoints in kubelet: `/metrics`, `/metrics/resource`, `/metrics/probe` and `/metrics/cadvisor`,
+all of which are exposed with a Prometheus style. The Metrics resource is capable of simulating endpoints with such style.
+
+To simulate a metric endpoint, first, you need to specify the RESTful `path` of the endpoint,
+which will be installed and exposed by the metric service of `kwok` at port `10247` after applied.
+The `path` must start with `/metrics`, otherwise, `kwok` will not install it.
+
+
+{{< hint "info" >}}
+Starting from metrics-server 0.7.0, it is allowed to specify the path to scrape metrics for a node.
+Specifically, metrics-server will check if a node has annotation `metrics.k8s.io/resource-metrics-path`
+and use it as the target metric scrape path.
+Combined with the Metric CR, the feature makes it possible to integrate `kwok` and metrics-server.
+For a fake node, by adding that annotation and setting its value to the `path`
+specified in a Metric resource, metrics-server will collect data from the endpoints exposed by `kwok` instead of
+scrapping from kubelet.
+{{< /hint >}}
+
+Besides, compared to kubelet, which only exposes the metric of the node it is located on, `kwok` needs to expose the
+metrics of all the fake nodes it manages. Instead of creating a separate Metric CR for each fake node, it is possible
+to bind all the metrics endpoints from different nodes into a single `path`. Metric CR allows for a built-in
+`{nodeName}` path parameter to be included in the `path` field. For example: `/metrics/nodes/{nodeName}/metrics/resource`.
+With `{nodeName}`, a single `path` is able to differentiate the metric data from different nodes.
+
+
+The `metrics` field are used to customize the return body of the installed metrics endpoint.
+
+The descriptions of each sub-field are available at [Metric API][Metric].
+For readers' convenience, we also mirror the documents here with some additional notes.
+
+`metrics` is a list of specific configuration items, with each corresponding to a Prometheus style metric:
+* `name` defines the metric name.
+* `labels` defines the metric labels, with each item corresponding to a specific metric label.
+ - `name` is a const string that provides the label name.
+ - `value` is represented as a CEL expression that dynamically determines the label value.
+ For example: you can use `node.metadata.name` to reference the node name as the label value.
+* `help` defines the help string of a metric.
+* `kind` defines the type of the metric: `counter`, `gauge`, or `histogram`.
+* `dimension` defines where the data comes from. It could be `node`, `pod`, or `container`.
+* `value` is a CEL expression that defines the metric value if `kind` is `counter` or `gauge`.
+ Please refer to [CEL expressions in `kwok`] for more detailed instructions that might be helpful to simulate the metric value.
+* `buckets` is exclusively for customizing the data of the metric of kind `histogram`.
+ - `le`, which defines the histogram bucket’s upper threshold, has the same meaning as the one of Prometheus histogram bucket.
+ That is, each bucket contains values less than or equal to `le`.
+ - `value` is a CEL expression that provides the value of the bucket.
+ - `hidden` indicates whether to show the bucket in the metric.
+ But the value of the bucket will be calculated and cumulated into the next bucket.
+
+Please refer to [Metrics for kubelet's "metrics/resource" endpoint][metrics resource endpoint] for a detailed example.
+
+
+## Out-of-box Metric Config
+
+`kwok` currently provides the [Metrics config][metrics resource endpoint] that is capable of
+simulating kubelet's `"metrics/resource"` endpoint.
+
+To integrate the simulated endpoint with metrics-server (required version >= 0.7.0), add the
+`"metrics.k8s.io/resource-metrics-path": "/metrics/nodes//metrics/resource"` annotation to the fake
+nodes managed by `kwok`.
+
+
+
+
+[configuration]: {{< relref "/docs/user/configuration" >}}
+[Metrics]: {{< relref "/docs/generated/apis" >}}#kwok.x-k8s.io/v1alpha1.Metrics
+[CEL expressions in `kwok`]: {{< relref "/docs/user/cel-expressions" >}}
+[metrics resource endpoint]: https://github.com/kubernetes-sigs/kwok/blob/main/kustomize/metrics/resource
+[ResourceUsage (ClusterResourceUsage)]: {{< relref "/docs/user/resource-usage-configuration" >}}
diff --git a/site/content/en/docs/user/resource-usage-configuration.md b/site/content/en/docs/user/resource-usage-configuration.md
new file mode 100644
index 000000000..bffe52dcc
--- /dev/null
+++ b/site/content/en/docs/user/resource-usage-configuration.md
@@ -0,0 +1,135 @@
+---
+title: ResourceUsage
+---
+
+# ResourceUsage Configuration
+
+{{< hint "info" >}}
+
+This document walks you through how to simulate the resource usage of pod(s).
+
+{{< /hint >}}
+
+## What is ResourceUsage?
+
+[ResourceUsage] is a [`kwok` Configuration][configuration] that allows users to define and simulate the resource usages of a single pod.
+
+The YAML below shows all the fields of a ResourceUsage resource:
+
+``` yaml
+kind: ResourceUsage
+apiVersion: kwok.x-k8s.io/v1alpha1
+metadata:
+ name:
+ namespace:
+spec:
+ usages:
+ - containers:
+ -
+ usage:
+ cpu:
+ value:
+ expression:
+ memory:
+ value:
+ expression:
+```
+
+To associate a ResourceUsage with a certain pod to be simulated, users must ensure `metadata.name` and `metadata.namespace`
+are inconsistent with the name and namespace of the target pod.
+
+The resource usages of a pod are specified via `usages` field.
+The `usages` field are organized by groups, with each corresponding to a collection of containers that shares a same resource usage simulation setting.
+Each group consists of a list of container names (`containers`) and the shared resource usage setting (`usage`).
+
+{{< hint "info" >}}
+If `containers` is not given in a group, the `usage` in that group will be applied to all containers of the target pod.
+{{< /hint >}}
+
+You can simply set a static [Quantity value] (`100Mi`, `1000m`, etc.) via `cpu.value` and `memory.value` to define the cpu and memory resource usage respectively.
+Besides, users are also allowed to provide a [CEL expression] via `expressions` to describe the resource usage more flexibly. For example,
+the following expression tries to extract the cpu resource usage from the pod's annotation if it has or use a default value.
+
+```yaml
+expression: |
+ "kwok.x-k8s.io/usage-cpu" in pod.metadata.annotations
+ ? Quantity(pod.metadata.annotations["kwok.x-k8s.io/usage-cpu"])
+ : Quantity("1m")
+```
+
+{{< hint "info" >}}
+1. `value` has higher priority than `expressions` if both are set.
+2. Quantity value must be explicitly wrapped by `Quantity` function in CEL expressions.
+{{< /hint >}}
+
+With CEL expressions, it is even possible to simulate resource usages dynamically. For example, the following expression
+yields memory usage that grows linearly with time.
+```yaml
+expression: (pod.SinceSecond() / 60.0) * Quantity("1Mi")
+```
+Please refer to [CEL expressions in `kwok`] for an exhausted list that may be helpful to configure dynamic resource usage.
+
+
+### ClusterResourceUsage
+
+In addition to simulating a single pod, users can also simulate the resource usage for multiple pods via [ClusterResourceUsage].
+
+The YAML below shows all the fields of a ClusterResourceUsage resource:
+
+``` yaml
+kind: ClusterResourceUsage
+apiVersion: kwok.x-k8s.io/v1alpha1
+metadata:
+ name:
+spec:
+ selector:
+ matchNamespaces:
+ -
+ matchNames:
+ -
+ usages:
+ - containers:
+ -
+ usage:
+ cpu:
+ value:
+ expression:
+ memory:
+ value:
+ expression:
+```
+Compared to ResourceUsage, whose `metadata.name` and `metadata.namespace` are required to match the associated pod,
+ClusterResourceUsage has an additional `selector` field for specifying the target pods to be simulated.
+`matchNamespaces` and `matchNames` are both represented as list,which are designed to take pod collections by different levels:
+
+1. If `matchNamespaces` is empty, ClusterResourceUsage will be applied to all pods that are managed by `kwok` and whose names listed in `matchNames`.
+2. If `matchNames` is empty, ClusterResourceUsage will be applied to all pods managed by `kwok` and under namespaces listed in `matchNamespaces`.
+3. If `matchNames` and `matchNamespaces` are both unset, ClusterResourceUsage will be applied to all pods that `kwok` manages.
+
+The `usages` field of ClusterResourceUsage has the same semantic with the one in ResourceUsage.
+
+Please refer to [pod resource usage from annotation] for a concrete example.
+
+## Where to get the simulation data?
+
+The resource usages defined in ResourceUsage and ClusterResourceUsage resources can be fetched from the metric service of `kwok` at port `10247` with path `/metrics/nodes/{nodeName}/metrics/resource`,
+where `{nodeName}` is the name of the fake node that the pod is scheduled to.
+The returned metrics are similar to the response from kubelet's `/metrics/resource` endpoint.
+
+Please refer to [`kwok` Metric][Metric] about how to integrate `kwok` simulated metrics endpoints with metrics-server.
+
+## Dependencies
+
+ResourceUsage or ClusterResourceUsage only takes effect when the [Metric] feature is also enabled and
+[the default Metric resource] that simulates kubelet's `/metrics/resource` endpoint is applied.
+
+
+[configuration]: {{< relref "/docs/user/configuration" >}}
+[ResourceUsage]: {{< relref "/docs/generated/apis" >}}#kwok.x-k8s.io/v1alpha1.ResourceUsage
+[ClusterResourceUsage]: {{< relref "/docs/generated/apis" >}}#kwok.x-k8s.io/v1alpha1.ClusterResourceUsage
+[Quantity value]: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/#resource-units-in-kubernetes
+[CEL expression]: https://github.com/google/cel-spec/blob/master/doc/langdef.md
+[Metric]: {{< relref "/docs/user/metrics-configuration" >}}
+[the default Metric resource]: https://github.com/kubernetes-sigs/kwok/blob/main/kustomize/metrics/resource
+[pod resource usage from annotation]: https://github.com/kubernetes-sigs/kwok/blob/main/kustomize/metrics/usage/usage-from-annotation.yaml
+[CEL expressions in `kwok`]: {{< relref "/docs/user/cel-expressions" >}}
diff --git a/site/content/en/docs/user/stages-configuration.md b/site/content/en/docs/user/stages-configuration.md
index a469c6293..12bf7a113 100644
--- a/site/content/en/docs/user/stages-configuration.md
+++ b/site/content/en/docs/user/stages-configuration.md
@@ -65,8 +65,9 @@ users can specify the conditions that need to be met for the stage to be applied
and the changes that will be made to the resource when the stage is applied.
The `next` field allows users to define the new status of the resource using the `statusTemplate` field, and even `delete` the resource.
-`statusTemplate` and `delete` are the two fundamental fields in `next` that respectively represent the two basic phases
-of resource lifecycle simulation: status update and resource deletion.
+`statusTemplate` and `delete` are the two fundamental fields in `next` that respectively represent the two basic phases of resource lifecycle simulation: status update and resource deletion.
+`statusTemplate` provides a way to define resource status based on go template rendering. Please see
+[go template in `kwok`] for more detailed instructions.
`delete: true` has higher priority than a non-empty `statusTemplate`, which means `kwok` will delete the resource
rather than update its status if both are set.
@@ -257,3 +258,4 @@ This example shows how to configure the simplest and fastest stages of Pod resou
[Stage API]: {{< relref "/docs/generated/apis" >}}#kwok.x-k8s.io/v1alpha1.Stage
[Resource Lifecycle Simulation Controller]: {{< relref "/docs/design/architecture" >}}
[How Delay is Calculated]: {{< relref "/docs/user/stages-configuration#how-delay-is-calculated" >}}
+[go template in `kwok`]: {{< relref "/docs/user/go-template" >}}
diff --git a/site/static/img/demo/resource-usage.svg b/site/static/img/demo/resource-usage.svg
new file mode 120000
index 000000000..2492dab3c
--- /dev/null
+++ b/site/static/img/demo/resource-usage.svg
@@ -0,0 +1 @@
+../../../../demo/resource-usage.svg
\ No newline at end of file