A complete example
Categories:
In the current documentation page, we will go through every step a ChallMaker will encounter, from the concept to players deploying instances in production.
flowchart LR Concept --> Build Build --> Scenario Scenario --> Pack Pack --> Deploy subgraph Build B1["Docker image"] end subgraph Scenario S1["Pulumi.yaml"] subgraph ExposedMonopod S2["main"] end S1 -.-> S2 end subgraph Pack P1["scenario.zip"] end
The concept
Imagine you are a ChallMaker who wants to challenge players on web application pentest.
The challenge is an application that require a licence to unlock functionalities, especially one that can read files vulnerable to local file inclusion. The initial access is due to an admin
:admin
account. The state of the application will be updated once the licence check is bypassed.
Its artistic direction is considered out of scope for now, but you’ll find this example all along our documentation !
flowchart LR A["Web app access"] A --> |"Initial access with admin:admin"| B B["Administrator account"] B --> |"License validation"| C C["Unlock vulnerable feature"] C --> |"Local File Inclusion"| D D["Read flag"]
Obviously, you don’t want players to impact others during their journey: Challenge on Demand is a solution.
Build the challenge
Using your bests software engineering skills, you conceive the application in the language of your choice, with the framework you are used to. You quickly test it, everthing behaves as expected, so you build a Write-Up for acceptance by reviewers.
This challenge is then packed into a Docker image: account/challenge:latest
We will then want to deploy this Docker image for every source that wants it.
Construct the scenario
To deploy this scenario, we don’t have big needs: one container, and a Kubernetes cluster. We’ll use the Kubernetes ExposedMonopod to ease its deployment.
First of all, we create the Pulumi.yaml
file to handle the scenario.
We write it to handle a pre-compiled binary of the scenario, for better performances.
Pulumi.yaml
name: stateful-webapp
description: The scenario to deploy the stateful web app challenge.
runtime:
name: go
options:
binary: ./main
Create the Go module using go mod init example
.
Then, we write the scenario file.
main.go
package main
import (
"github.com/ctfer-io/chall-manager/sdk"
"github.com/ctfer-io/chall-manager/sdk/kubernetes"
"github.com/pulumi/pulumi/sdk/v3/go/pulumi"
)
func main() {
sdk.Run(func(req *sdk.Request, resp *sdk.Response, opts ...pulumi.ResourceOption) error {
cm, err := kubernetes.NewExposedMonopod(req.Ctx, &kubernetes.ExposedMonopodArgs{
Image: pulumi.String("account/challenge:latest"), // challenge Docker image
Port: pulumi.Int(8080), // pod listens on port 8080
ExposeType: kubernetes.ExposeIngress, // expose the challenge through an ingress (HTTP)
Hostname: pulumi.String("brefctf.ctfer.io"), // CTF hostname
Identity: pulumi.String(req.Config.Identity), // identity will be prepended to hostname
}, opts...)
if err != nil {
return err
}
resp.ConnectionInfo = pulumi.Sprintf("curl -v https://%s", cm.URL) // a simple web server
return nil
})
}
Download the required dependencies using go mod tidy
.
To test it you can open a terminal and execute pulumi up
. It requires your host machine to have a kubeconfig
or a ServiceAccount
token in its filesystem, i.e. you are able to execute commands like kubectl get pods -A
.
Finally, compile using CGO_ENABLED=0 go build ./main main.go
.
Send it to chall-manager
The challenge is ready to be deployed. To give those information to chall-manager, you have to build the scenario zip archive.
As the scenario has been compiled, we only have to archive the Pulumi.yaml
and main
files.
zip -r scenario.zip Pulumi.yaml main
Then, you have to create the challenge (e.g. some-challenge
) in chall-manager. You can do it using the gRPC API or the HTTP gateway.
We’ll use chall-manager-cli
to do so easily.
chall-manager-cli --url chall-manager:8080 challenge create \
--id "some-challenge" \
--file scenario.zip
Now, chall-manager is able to deploy our challenge for players.
Deploy instances
To deploy instances, we’ll mock a player (e.g. mocking-bird
) Challenge on Demand request using chall-manager-cli
.
In reality, it would be to the CTF platform to handle the previous step and this one, but it is considered out of scope.
chall-manager-cli --url chall-manager:8080 instance create \
--challenge_id "some-challenge" \
--source_id "mocking-bird"
This will return us the connection information to our instance of the challenge.
Feedback
Was this page helpful?