From 4f7df39edd3c023d355fc0d8fd938f9ea834ce91 Mon Sep 17 00:00:00 2001 From: qwerty287 <80460567+qwerty287@users.noreply.github.com> Date: Thu, 13 Jun 2024 19:31:54 +0200 Subject: [PATCH] Add 2.6 docs (#3786) --- docs/docs/20-usage/45-cron.md | 2 +- docs/docusaurus.config.ts | 14 +- docs/src/pages/versions.md | 1 + .../11-forges/10-overview.md | 13 - .../22-backends/40-kubernetes.md | 217 --- .../30-administration/40-encryption.md | 83 -- .../75-addons/00-overview.md | 45 - .../75-addons/20-creating-addons.md | 102 -- .../75-addons/_category_.yaml | 6 - docs/versioned_docs/version-2.3/40-cli.md | 1215 ----------------- .../{version-2.3 => version-2.6}/10-intro.md | 4 +- .../20-usage/10-intro.md | 0 .../20-usage/100-troubleshooting.md | 37 + .../15-terminology}/architecture.excalidraw | 0 .../20-usage/15-terminology}/architecture.svg | 0 .../20-usage/15-terminology}/index.md | 6 +- .../pipeline-workflow-step.excalidraw | 0 .../pipeline-workflow-step.svg | 0 .../20-usage/20-workflow-syntax.md | 207 +-- .../20-usage/25-workflows.md | 4 +- .../20-usage/30-matrix-workflows.md | 2 +- .../20-usage/40-secrets.md | 43 +- .../20-usage/41-registries.md | 4 + .../20-usage/45-cron.md | 4 +- .../20-usage/50-environment.md | 12 +- .../51-plugins/20-creating-plugins.md | 23 +- .../20-usage/51-plugins/51-overview.md} | 0 .../20-usage/51-plugins/_category_.yaml | 0 .../20-usage/60-services.md | 0 .../20-usage/70-volumes.md | 2 +- .../version-2.6/20-usage/72-linter.md | 62 + .../20-usage/75-project-settings.md} | 19 +- .../20-usage/80-badges.md | 0 .../20-usage/90-advanced-usage.md | 0 .../20-usage/_category_.yaml | 0 .../20-usage/cron-settings.png | Bin .../20-usage/linter-warnings-errors.png | Bin 0 -> 116049 bytes .../20-usage/project-settings.png | Bin .../20-usage/repo-list.png | Bin .../00-deployment/00-overview.md | 24 +- .../00-deployment/10-docker-compose.md | 2 +- .../00-deployment/20-kubernetes.md | 0 .../00-deployment/30-nixos.md | 4 +- .../00-deployment/_category_.yaml | 0 .../30-administration/10-server-config.md | 66 +- .../100-external-configuration-api.md | 0 .../30-administration/11-forges/100-addon.md | 68 + .../11-forges/11-overview.md | 13 + .../30-administration/11-forges/20-github.md | 6 + .../30-administration/11-forges/30-gitea.md | 14 +- .../30-administration/11-forges/35-forgejo.md | 97 ++ .../30-administration/11-forges/40-gitlab.md | 0 .../11-forges/50-bitbucket.md | 2 +- .../11-forges/60-bitbucket_datacenter.md | 98 ++ .../11-forges/_category_.yaml | 0 .../11-forges/bitbucket_oauth.png | Bin .../11-forges/bitbucket_permissions.png | Bin .../11-forges/gitea_oauth.gif | Bin .../11-forges/github_oauth.png | Bin .../30-administration/15-agent-config.md | 6 - .../22-backends/10-docker.md | 0 .../30-administration/22-backends/20-local.md | 24 +- .../22-backends/40-kubernetes.md | 305 +++++ .../22-backends/50-custom-backends.md | 23 + .../22-backends/_category_.yaml | 0 .../30-administration/30-database.md | 0 .../30-administration/60-ssl.md | 2 +- .../30-administration/70-proxy.md | 2 +- .../30-administration/80-autoscaler.md | 0 .../30-administration/90-prometheus.md | 0 .../30-administration/_category_.yaml | 0 .../30-administration/new-agent-connected.png | Bin .../30-administration/new-agent-created.png | Bin .../new-agent-registration.png | Bin docs/versioned_docs/version-2.6/40-cli.md | 594 ++++++++ docs/versioned_docs/version-2.6/50-about.md | 18 + .../91-migrations.md | 16 +- .../92-awesome.md | 17 +- .../92-development/01-getting-started.md | 9 +- .../92-development/02-core-ideas.md | 26 + .../92-development/03-ui.md | 0 .../92-development/04-docs.md | 0 .../92-development/05-architecture.md | 4 +- .../92-development/06-guides.md | 1 - .../92-development/07-translations.md | 0 .../92-development/08-swagger.md | 2 +- .../92-development/_category_.yaml | 0 .../92-development/ui-proxy.svg | 0 .../92-development/vscode-debug.png | Bin .../92-development/vscode-run-test.png | Bin .../woodpecker-architecture.png | Bin .../woodpecker.png | Bin ...idebars.json => version-2.6-sidebars.json} | 0 docs/versions.json | 2 +- 94 files changed, 1571 insertions(+), 2001 deletions(-) delete mode 100644 docs/versioned_docs/version-2.3/30-administration/11-forges/10-overview.md delete mode 100644 docs/versioned_docs/version-2.3/30-administration/22-backends/40-kubernetes.md delete mode 100644 docs/versioned_docs/version-2.3/30-administration/40-encryption.md delete mode 100644 docs/versioned_docs/version-2.3/30-administration/75-addons/00-overview.md delete mode 100644 docs/versioned_docs/version-2.3/30-administration/75-addons/20-creating-addons.md delete mode 100644 docs/versioned_docs/version-2.3/30-administration/75-addons/_category_.yaml delete mode 100644 docs/versioned_docs/version-2.3/40-cli.md rename docs/versioned_docs/{version-2.3 => version-2.6}/10-intro.md (96%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/10-intro.md (100%) create mode 100644 docs/versioned_docs/version-2.6/20-usage/100-troubleshooting.md rename docs/versioned_docs/{version-2.3/20-usage/15-terminiology => version-2.6/20-usage/15-terminology}/architecture.excalidraw (100%) rename docs/versioned_docs/{version-2.3/20-usage/15-terminiology => version-2.6/20-usage/15-terminology}/architecture.svg (100%) rename docs/versioned_docs/{version-2.3/20-usage/15-terminiology => version-2.6/20-usage/15-terminology}/index.md (92%) rename docs/versioned_docs/{version-2.3/20-usage/15-terminiology => version-2.6/20-usage/15-terminology}/pipeline-workflow-step.excalidraw (100%) rename docs/versioned_docs/{version-2.3/20-usage/15-terminiology => version-2.6/20-usage/15-terminology}/pipeline-workflow-step.svg (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/20-workflow-syntax.md (79%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/25-workflows.md (96%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/30-matrix-workflows.md (99%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/40-secrets.md (75%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/41-registries.md (94%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/45-cron.md (88%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/50-environment.md (96%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/51-plugins/20-creating-plugins.md (81%) rename docs/versioned_docs/{version-2.3/20-usage/51-plugins/10-overview.md => version-2.6/20-usage/51-plugins/51-overview.md} (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/51-plugins/_category_.yaml (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/60-services.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/70-volumes.md (93%) create mode 100644 docs/versioned_docs/version-2.6/20-usage/72-linter.md rename docs/versioned_docs/{version-2.3/20-usage/71-project-settings.md => version-2.6/20-usage/75-project-settings.md} (86%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/80-badges.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/90-advanced-usage.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/_category_.yaml (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/cron-settings.png (100%) create mode 100644 docs/versioned_docs/version-2.6/20-usage/linter-warnings-errors.png rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/project-settings.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/20-usage/repo-list.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/00-deployment/00-overview.md (77%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/00-deployment/10-docker-compose.md (95%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/00-deployment/20-kubernetes.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/00-deployment/30-nixos.md (95%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/00-deployment/_category_.yaml (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/10-server-config.md (92%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/100-external-configuration-api.md (100%) create mode 100644 docs/versioned_docs/version-2.6/30-administration/11-forges/100-addon.md create mode 100644 docs/versioned_docs/version-2.6/30-administration/11-forges/11-overview.md rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/20-github.md (93%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/30-gitea.md (84%) create mode 100644 docs/versioned_docs/version-2.6/30-administration/11-forges/35-forgejo.md rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/40-gitlab.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/50-bitbucket.md (97%) create mode 100644 docs/versioned_docs/version-2.6/30-administration/11-forges/60-bitbucket_datacenter.md rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/_category_.yaml (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/bitbucket_oauth.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/bitbucket_permissions.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/gitea_oauth.gif (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/11-forges/github_oauth.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/15-agent-config.md (98%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/22-backends/10-docker.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/22-backends/20-local.md (54%) create mode 100644 docs/versioned_docs/version-2.6/30-administration/22-backends/40-kubernetes.md create mode 100644 docs/versioned_docs/version-2.6/30-administration/22-backends/50-custom-backends.md rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/22-backends/_category_.yaml (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/30-database.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/60-ssl.md (98%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/70-proxy.md (99%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/80-autoscaler.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/90-prometheus.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/_category_.yaml (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/new-agent-connected.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/new-agent-created.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/30-administration/new-agent-registration.png (100%) create mode 100644 docs/versioned_docs/version-2.6/40-cli.md create mode 100644 docs/versioned_docs/version-2.6/50-about.md rename docs/versioned_docs/{version-2.3 => version-2.6}/91-migrations.md (92%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-awesome.md (80%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/01-getting-started.md (92%) create mode 100644 docs/versioned_docs/version-2.6/92-development/02-core-ideas.md rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/03-ui.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/04-docs.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/05-architecture.md (92%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/06-guides.md (92%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/07-translations.md (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/08-swagger.md (97%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/_category_.yaml (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/ui-proxy.svg (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/vscode-debug.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/vscode-run-test.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/92-development/woodpecker-architecture.png (100%) rename docs/versioned_docs/{version-2.3 => version-2.6}/woodpecker.png (100%) rename docs/versioned_sidebars/{version-2.3-sidebars.json => version-2.6-sidebars.json} (100%) diff --git a/docs/docs/20-usage/45-cron.md b/docs/docs/20-usage/45-cron.md index bbf23bdf0..95ee8202e 100644 --- a/docs/docs/20-usage/45-cron.md +++ b/docs/docs/20-usage/45-cron.md @@ -23,7 +23,7 @@ To configure cron jobs you need at least push access to the repository. ![cron settings](./cron-settings.png) - The supported schedule syntax can be found at . If you need general understanding of the cron syntax is a good place to start and experiment. + The supported schedule syntax can be found at . If you need general understanding of the cron syntax is a good place to start and experiment. Examples: `@every 5m`, `@daily`, `0 30 * * * *` ... diff --git a/docs/docusaurus.config.ts b/docs/docusaurus.config.ts index 5bbb8d640..5057fa018 100644 --- a/docs/docusaurus.config.ts +++ b/docs/docusaurus.config.ts @@ -243,25 +243,25 @@ const config: Config = { sidebarPath: require.resolve('./sidebars.js'), editUrl: 'https://github.com/woodpecker-ci/woodpecker/edit/main/docs/', includeCurrentVersion: true, - lastVersion: '2.5', + lastVersion: '2.6', onlyIncludeVersions: - process.env.NODE_ENV === 'development' ? ['current', '2.5'] : ['current', '2.5', '2.4', '2.3', '1.0'], + process.env.NODE_ENV === 'development' ? ['current', '2.6'] : ['current', '2.6', '2.5', '2.4', '1.0'], versions: { current: { label: 'Next 🚧', banner: 'unreleased', }, + '2.6': { + label: '2.6.x', + }, '2.5': { - label: '2.5.x', + label: '2.5.x 💀', + banner: 'unmaintained', }, '2.4': { label: '2.4.x 💀', banner: 'unmaintained', }, - '2.3': { - label: '2.3.x 💀', - banner: 'unmaintained', - }, '1.0': { label: '1.0.x 💀', banner: 'unmaintained', diff --git a/docs/src/pages/versions.md b/docs/src/pages/versions.md index 98f48f1e8..6e7fbffc4 100644 --- a/docs/src/pages/versions.md +++ b/docs/src/pages/versions.md @@ -33,6 +33,7 @@ Here you can find documentation for previous versions of Woodpecker. | | | | | ------- | ---------- | ------------------------------------------------------------------------------------- | +| 2.5.0 | 2024-06-01 | [Documentation](https://github.com/woodpecker-ci/woodpecker/tree/v2.5.0/docs/docs/) | | 2.4.1 | 2024-03-20 | [Documentation](https://github.com/woodpecker-ci/woodpecker/tree/v2.4.1/docs/docs/) | | 2.4.0 | 2024-03-19 | [Documentation](https://github.com/woodpecker-ci/woodpecker/tree/v2.4.0/docs/docs/) | | 2.3.0 | 2024-01-31 | [Documentation](https://github.com/woodpecker-ci/woodpecker/tree/v2.3.0/docs/docs/) | diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/10-overview.md b/docs/versioned_docs/version-2.3/30-administration/11-forges/10-overview.md deleted file mode 100644 index bacce1635..000000000 --- a/docs/versioned_docs/version-2.3/30-administration/11-forges/10-overview.md +++ /dev/null @@ -1,13 +0,0 @@ -# Forges - -## Supported features - -| Feature | [GitHub](github/) | [Gitea / Forgejo](gitea/) | [Gitlab](gitlab/) | [Bitbucket](bitbucket/) | -| ------------------------------------------------------------- | :----------------: | :-----------------------: | :----------------: | :---------------------: | -| Event: Push | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | -| Event: Tag | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | -| Event: Pull-Request | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | -| Event: Release | :white_check_mark: | :white_check_mark: | :white_check_mark: | :x: | -| Event: Deploy | :white_check_mark: | :x: | :x: | :x: | -| [Multiple workflows](../../20-usage/25-workflows.md) | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | -| [when.path filter](../../20-usage/20-workflow-syntax.md#path) | :white_check_mark: | :white_check_mark: | :white_check_mark: | :x: | diff --git a/docs/versioned_docs/version-2.3/30-administration/22-backends/40-kubernetes.md b/docs/versioned_docs/version-2.3/30-administration/22-backends/40-kubernetes.md deleted file mode 100644 index 2da6c6545..000000000 --- a/docs/versioned_docs/version-2.3/30-administration/22-backends/40-kubernetes.md +++ /dev/null @@ -1,217 +0,0 @@ ---- -toc_max_heading_level: 2 ---- - -# Kubernetes backend - -The kubernetes backend executes steps inside standalone pods. A temporary PVC is created for the lifetime of the pipeline to transfer files between steps. - -## Job specific configuration - -### Resources - -The kubernetes backend also allows for specifying requests and limits on a per-step basic, most commonly for CPU and memory. -We recommend to add a `resources` definition to all steps to ensure efficient scheduling. - -Here is an example definition with an arbitrary `resources` definition below the `backend_options` section: - -```yaml -steps: - - name: 'My kubernetes step' - image: alpine - commands: - - echo "Hello world" - backend_options: - kubernetes: - resources: - requests: - memory: 200Mi - cpu: 100m - limits: - memory: 400Mi - cpu: 1000m -``` - -### `serviceAccountName` - -Specify the name of the ServiceAccount which the build pod will mount. This serviceAccount must be created externally. -See the [kubernetes documentation](https://kubernetes.io/docs/concepts/security/service-accounts/) for more information on using serviceAccounts. - -### `nodeSelector` - -Specifies the label which is used to select the node on which the job will be executed. - -Labels defined here will be appended to a list which already contains `"kubernetes.io/arch"`. -By default `"kubernetes.io/arch"` is inferred from the agents' platform. One can override it by setting that label in the `nodeSelector` section of the `backend_options`. -Without a manual overwrite, builds will be randomly assigned to the runners and inherit their respective architectures. - -To overwrite this, one needs to set the label in the `nodeSelector` section of the `backend_options`. -A practical example for this is when running a matrix-build and delegating specific elements of the matrix to run on a specific architecture. -In this case, one must define an arbitrary key in the matrix section of the respective matrix element: - -```yaml -matrix: - include: - - NAME: runner1 - ARCH: arm64 -``` - -And then overwrite the `nodeSelector` in the `backend_options` section of the step(s) using the name of the respective env var: - -```yaml -[...] - backend_options: - kubernetes: - nodeSelector: - kubernetes.io/arch: "${ARCH}" -``` - -### `tolerations` - -When you use nodeSelector and the node pool is configured with Taints, you need to specify the Tolerations. Tolerations allow the scheduler to schedule pods with matching taints. -See the [kubernetes documentation](https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/) for more information on using tolerations. - -Example pipeline configuration: - -```yaml -steps: - - name: build - image: golang - commands: - - go get - - go build - - go test - backend_options: - kubernetes: - serviceAccountName: 'my-service-account' - resources: - requests: - memory: 128Mi - cpu: 1000m - limits: - memory: 256Mi - nodeSelector: - beta.kubernetes.io/instance-type: p3.8xlarge - tolerations: - - key: 'key1' - operator: 'Equal' - value: 'value1' - effect: 'NoSchedule' - tolerationSeconds: 3600 -``` - -### Volumes - -To mount volumes a persistent volume (PV) and persistent volume claim (PVC) are needed on the cluster which can be referenced in steps via the `volume:` option. -Assuming a PVC named "woodpecker-cache" exists, it can be referenced as follows in a step: - -```yaml -steps: - - name: "Restore Cache" - image: meltwater/drone-cache - volumes: - - woodpecker-cache:/woodpecker/src/cache - settings: - mount: - - "woodpecker-cache" - [...] -``` - -### `securityContext` - -Use the following configuration to set the `securityContext` for the pod/container running a given pipeline step: - -```yaml -steps: - - name: test - image: alpine - commands: - - echo Hello world - backend_options: - kubernetes: - securityContext: - runAsUser: 999 - runAsGroup: 999 - privileged: true - [...] -``` - -Note that the `backend_options.kubernetes.securityContext` object allows you to set both pod and container level security context options in one object. -By default, the properties will be set at the pod level. Properties that are only supported on the container level will be set there instead. So, the -configuration shown above will result in something like the following pod spec: - -```yaml -kind: Pod -spec: - securityContext: - runAsUser: 999 - runAsGroup: 999 - containers: - - name: wp-01hcd83q7be5ymh89k5accn3k6-0-step-0 - image: alpine - securityContext: - privileged: true - [...] -``` - -See the [kubernetes documentation](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/) for more information on using `securityContext`. - -## Tips and tricks - -### CRI-O - -CRI-O users currently need to configure the workspace for all workflows in order for them to run correctly. Add the following at the beginning of your configuration: - -```yaml -workspace: - base: '/woodpecker' - path: '/' -``` - -See [this issue](https://github.com/woodpecker-ci/woodpecker/issues/2510) for more details. - -## Configuration - -These env vars can be set in the `env:` sections of the agent. - -### `WOODPECKER_BACKEND_K8S_NAMESPACE` - -> Default: `woodpecker` - -The namespace to create worker pods in. - -### `WOODPECKER_BACKEND_K8S_VOLUME_SIZE` - -> Default: `10G` - -The volume size of the pipeline volume. - -### `WOODPECKER_BACKEND_K8S_STORAGE_CLASS` - -> Default: empty - -The storage class to use for the pipeline volume. - -### `WOODPECKER_BACKEND_K8S_STORAGE_RWX` - -> Default: `true` - -Determines if `RWX` should be used for the pipeline volume's [access mode](https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes). If false, `RWO` is used instead. - -### `WOODPECKER_BACKEND_K8S_POD_LABELS` - -> Default: empty - -Additional labels to apply to worker pods. Must be a YAML object, e.g. `{"example.com/test-label":"test-value"}`. - -### `WOODPECKER_BACKEND_K8S_POD_ANNOTATIONS` - -> Default: empty - -Additional annotations to apply to worker pods. Must be a YAML object, e.g. `{"example.com/test-annotation":"test-value"}`. - -### `WOODPECKER_BACKEND_K8S_SECCTX_NONROOT` - -> Default: `false` - -Determines if containers must be required to run as non-root users. diff --git a/docs/versioned_docs/version-2.3/30-administration/40-encryption.md b/docs/versioned_docs/version-2.3/30-administration/40-encryption.md deleted file mode 100644 index f630711d8..000000000 --- a/docs/versioned_docs/version-2.3/30-administration/40-encryption.md +++ /dev/null @@ -1,83 +0,0 @@ -# Secrets encryption - -:::danger -Secrets encryption is currently broken and therefore disabled by default. It will be fixed in an upcoming release. - -Check: - -- and -- - -::: - -By default, Woodpecker does not encrypt secrets in its database. You can enable encryption -using simple AES key or more advanced [Google TINK](https://developers.google.com/tink) encryption. - -## Common - -### Enabling secrets encryption - -To enable secrets encryption and encrypt all existing secrets in database set -`WOODPECKER_ENCRYPTION_KEY`, `WOODPECKER_ENCRYPTION_KEY_FILE` or `WOODPECKER_ENCRYPTION_TINK_KEYSET_PATH` environment -variable depending on encryption method of your choice. - -After encryption is enabled you will be unable to start Woodpecker server without providing valid encryption key! - -### Disabling encryption and decrypting all secrets - -To disable secrets encryption and decrypt database you need to start server with valid -`WOODPECKER_ENCRYPTION_KEY` or `WOODPECKER_ENCRYPTION_TINK_KEYSET_FILE` environment variable set depending on -enabled encryption method, and `WOODPECKER_ENCRYPTION_DISABLE` set to true. - -After secrets was decrypted server will proceed working in unencrypted mode. You will not need to use "disable encryption" -variable or encryption keys to start server anymore. - -## AES - -Simple AES encryption. - -### Configuration - -You can manage encryption on server using these environment variables: - -- `WOODPECKER_ENCRYPTION_KEY` - encryption key -- `WOODPECKER_ENCRYPTION_KEY_FILE` - file to read encryption key from -- `WOODPECKER_ENCRYPTION_DISABLE` - disable encryption flag used to decrypt all data on server - -## TINK - -TINK uses AEAD encryption instead of simple AES and supports key rotation. - -### Configuration - -You can manage encryption on server using these two environment variables: - -- `WOODPECKER_ENCRYPTION_TINK_KEYSET_FILE` - keyset filepath -- `WOODPECKER_ENCRYPTION_DISABLE` - disable encryption flag used to decrypt all data on server - -### Encryption keys - -You will need plaintext AEAD-compatible Google TINK keyset to encrypt your data. - -To generate it and then rotate keys if needed, install `tinkey`([installation guide](https://developers.google.com/tink/install-tinkey)) - -Keyset contains one or more keys, used to encrypt or decrypt your data, and primary key ID, used to determine which key -to use while encrypting new data. - -Keyset generation example: - -```bash -tinkey create-keyset --key-template AES256_GCM --out-format json --out keyset.json -``` - -### Key rotation - -Use `tinkey` to rotate encryption keys in your existing keyset: - -```bash -tinkey rotate-keyset --in keyset_v1.json --out keyset_v2.json --key-template AES256_GCM -``` - -Then you just need to replace server keyset file with the new one. At the moment server detects new encryption -keyset it will re-encrypt all existing secrets with the new key, so you will be unable to start server with previous -keyset anymore. diff --git a/docs/versioned_docs/version-2.3/30-administration/75-addons/00-overview.md b/docs/versioned_docs/version-2.3/30-administration/75-addons/00-overview.md deleted file mode 100644 index 485f9ce84..000000000 --- a/docs/versioned_docs/version-2.3/30-administration/75-addons/00-overview.md +++ /dev/null @@ -1,45 +0,0 @@ -# Addons - -:::warning -Addons are still experimental. Their implementation can change and break at any time. -::: - -:::danger -You need to trust the author of the addons you use. Depending on their type, addons can access forge authentication codes, your secrets or other sensitive information. -::: - -To adapt Woodpecker to your needs beyond the [configuration](../10-server-config.md), Woodpecker has its own **addon** system, built ontop of [Go's internal plugin system](https://go.dev/pkg/plugin). - -Addons can be used for: - -- Forges -- Agent backends -- Config services -- Secret services -- Environment services -- Registry services - -## Restrictions - -Addons are restricted by how Go plugins work. This includes the following restrictions: - -- only supported on Linux, FreeBSD, and macOS -- addons must have been built for the correct Woodpecker version. If an addon is not provided specifically for this version, you likely won't be able to use it. - -## Usage - -To use an addon, download the addon version built for your Woodpecker version. Then, you can add the following to your configuration: - -```ini -WOODPECKER_ADDONS=/path/to/your/addon/file.so -``` - -In case you run Woodpecker as container, you probably want to mount the addon binaries to `/opt/addons/`. - -You can list multiple addons, Woodpecker will automatically determine their type. If you specify multiple addons with the same type, only the first one will be used. - -Using an addon always overwrites Woodpecker's internal setup. This means, that a forge addon will be used if specified, no matter what's configured for the forges natively supported by Woodpecker. - -### Bug reports - -If you experience bugs, please check which component has the issue. If it's the addon, **do not raise an issue in the main repository**, but rather use the separate addon repositories. To check which component is responsible for the bug, look at the logs. Logs from addons are marked with a special field `addon` containing their addon file name. diff --git a/docs/versioned_docs/version-2.3/30-administration/75-addons/20-creating-addons.md b/docs/versioned_docs/version-2.3/30-administration/75-addons/20-creating-addons.md deleted file mode 100644 index 0fd6d2fe0..000000000 --- a/docs/versioned_docs/version-2.3/30-administration/75-addons/20-creating-addons.md +++ /dev/null @@ -1,102 +0,0 @@ -# Creating addons - -Addons are written in Go. - -## Writing your code - -An addon consists of two variables/functions in Go. - -1. The `Type` variable. Specifies the type of the addon and must be directly accessed from `shared/addons/types/types.go`. -2. The `Addon` function which is the main point of your addon. - This function takes the `zerolog` logger you should use to log errors, warnings, etc. as argument. - - It returns two values: - - 1. The actual addon. For type reference see [table below](#return-types). - 2. An error. If this error is not `nil`, Woodpecker exits. - -Directly import Woodpecker's Go package (`go.woodpecker-ci.org/woodpecker/woodpecker/v2`) and use the interfaces and types defined there. - -### Return types - -| Addon type | Return type | -| -------------------- | -------------------------------------------------------------------------------- | -| `Forge` | `"go.woodpecker-ci.org/woodpecker/woodpecker/v2/server/forge".Forge` | -| `Backend` | `"go.woodpecker-ci.org/woodpecker/woodpecker/v2/pipeline/backend/types".Backend` | -| `ConfigService` | `"go.woodpecker-ci.org/woodpecker/v2/server/plugins/config".Extension` | -| `SecretService` | `"go.woodpecker-ci.org/woodpecker/v2/server/model".SecretService` | -| `EnvironmentService` | `"go.woodpecker-ci.org/woodpecker/v2/server/model".EnvironmentService` | -| `RegistryService` | `"go.woodpecker-ci.org/woodpecker/v2/server/model".RegistryService` | - -### Using configurations - -If you write a plugin for the server (`Forge` and the services), you can access the server config. - -Therefore, use the `"go.woodpecker-ci.org/woodpecker/v2/server".Config` variable. - -:::warning -The config is not available when your addon is initialized, i.e., the `Addon` function is called. -Only use the config in the interface methods. -::: - -## Compiling - -After you write your addon code, compile your addon: - -```sh -go build -buildmode plugin -``` - -The output file is your addon that is now ready to be used. - -## Restrictions - -Addons must directly depend on Woodpecker's core (`go.woodpecker-ci.org/woodpecker/woodpecker/v2`). -The addon must have been built with **exactly the same code** as the Woodpecker instance you'd like to use it on. This means: If you build your addon with a specific commit from Woodpecker `next`, you can likely only use it with the Woodpecker version compiled from this commit. -Also, if you change something inside Woodpecker without committing, it might fail because you need to recompile your addon with this code first. - -In addition to this, addons are only supported on Linux, FreeBSD, and macOS. - -:::info -It is recommended to at least support the latest version of Woodpecker. -::: - -### Compile for different versions - -As long as there are no changes to Woodpecker's interfaces, -or they are backwards-compatible, you can compile the addon for multiple versions -by changing the version of `go.woodpecker-ci.org/woodpecker/woodpecker/v2` using `go get` before compiling. - -## Logging - -The entrypoint receives a `zerolog.Logger` as input. **Do not use any other logging solution.** This logger follows the configuration of the Woodpecker instance and adds a special field `addon` to the log entries which allows users to find out which component is writing the log messages. - -## Example structure - -```go -package main - -import ( - "context" - "net/http" - - "github.com/rs/zerolog" - "go.woodpecker-ci.org/woodpecker/v2/server/forge" - forge_types "go.woodpecker-ci.org/woodpecker/v2/server/forge/types" - "go.woodpecker-ci.org/woodpecker/v2/server/model" - addon_types "go.woodpecker-ci.org/woodpecker/v2/shared/addon/types" -) - -var Type = addon_types.TypeForge - -func Addon(logger zerolog.Logger) (forge.Forge, error) { - logger.Info().Msg("hello world from addon") - return &config{l: logger}, nil -} - -type config struct { - l zerolog.Logger -} - -// In this case, `config` must implement `forge.Forge`. You must directly use Woodpecker's packages - see imports above. -``` diff --git a/docs/versioned_docs/version-2.3/30-administration/75-addons/_category_.yaml b/docs/versioned_docs/version-2.3/30-administration/75-addons/_category_.yaml deleted file mode 100644 index 4cd7380c5..000000000 --- a/docs/versioned_docs/version-2.3/30-administration/75-addons/_category_.yaml +++ /dev/null @@ -1,6 +0,0 @@ -label: 'Addons' -collapsible: true -collapsed: true -link: - type: 'doc' - id: 'overview' diff --git a/docs/versioned_docs/version-2.3/40-cli.md b/docs/versioned_docs/version-2.3/40-cli.md deleted file mode 100644 index d8001ca97..000000000 --- a/docs/versioned_docs/version-2.3/40-cli.md +++ /dev/null @@ -1,1215 +0,0 @@ -# CLI - -# NAME - -woodpecker-cli - command line utility - -# SYNOPSIS - -woodpecker-cli - -``` -[--log-file]=[value] -[--log-level]=[value] -[--nocolor] -[--pretty] -[--server|-s]=[value] -[--token|-t]=[value] -``` - -**Usage**: - -``` -woodpecker-cli [GLOBAL OPTIONS] command [COMMAND OPTIONS] [ARGUMENTS...] -``` - -# GLOBAL OPTIONS - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - - -# COMMANDS - -## pipeline - -manage pipelines - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### ls - -show pipeline history - -**--branch**="": branch filter - -**--event**="": event filter - -**--format**="": format output (default: Pipeline #{{ .Number }}  -Status: {{ .Status }} -Event: {{ .Event }} -Commit: {{ .Commit }} -Branch: {{ .Branch }} -Ref: {{ .Ref }} -Author: {{ .Author }} {{ if .Email }}<{{.Email}}>{{ end }} -Message: {{ .Message }} -) - -**--limit**="": limit the list size (default: 0) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--status**="": status filter - -**--token, -t**="": server auth token - -### last - -show latest pipeline details - -**--branch**="": branch name (default: main) - -**--format**="": format output (default: Number: {{ .Number }} -Status: {{ .Status }} -Event: {{ .Event }} -Commit: {{ .Commit }} -Branch: {{ .Branch }} -Ref: {{ .Ref }} -Message: {{ .Message }} -Author: {{ .Author }} -) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### logs - -show pipeline logs - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### info - -show pipeline details - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### stop - -stop a pipeline - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### start - -start a pipeline - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### approve - -approve a pipeline - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### decline - -decline a pipeline - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### queue - -show pipeline queue - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### ps - -show pipeline steps - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### create - -create new pipeline - -**--branch**="": branch to create pipeline from - -**--format**="": format output (default: Pipeline #{{ .Number }}  -Status: {{ .Status }} -Event: {{ .Event }} -Commit: {{ .Commit }} -Branch: {{ .Branch }} -Ref: {{ .Ref }} -Author: {{ .Author }} {{ if .Email }}<{{.Email}}>{{ end }} -Message: {{ .Message }} -) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -**--var**="": key=value - -## log - -manage logs - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### purge - -purge a log - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -## deploy - -deploy code - -**--branch**="": branch filter (default: main) - -**--event**="": event filter (default: push) - -**--format**="": format output (default: Number: {{ .Number }} -Status: {{ .Status }} -Commit: {{ .Commit }} -Branch: {{ .Branch }} -Ref: {{ .Ref }} -Message: {{ .Message }} -Author: {{ .Author }} -Target: {{ .Deploy }} -) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--param, -p**="": custom parameters to be injected into the step environment. Format: KEY=value - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--status**="": status filter (default: success) - -**--token, -t**="": server auth token - -## exec - -execute a local pipeline - -**--backend-docker-api-version**="": the version of the API to reach, leave empty for latest. - -**--backend-docker-cert**="": path to load the TLS certificates for connecting to docker server - -**--backend-docker-host**="": path to docker socket or url to the docker server - -**--backend-docker-ipv6**: backend docker enable IPV6 - -**--backend-docker-network**="": backend docker network - -**--backend-docker-tls-verify**: enable or disable TLS verification for connecting to docker server - -**--backend-docker-volumes**="": backend docker volumes (comma separated) - -**--backend-engine**="": backend engine to run pipelines on (default: auto-detect) - -**--backend-http-proxy**="": if set, pass the environment variable down as "HTTP_PROXY" to steps - -**--backend-https-proxy**="": if set, pass the environment variable down as "HTTPS_PROXY" to steps - -**--backend-k8s-namespace**="": backend k8s namespace (default: woodpecker) - -**--backend-k8s-pod-annotations**="": backend k8s additional worker pod annotations - -**--backend-k8s-pod-labels**="": backend k8s additional worker pod labels - -**--backend-k8s-storage-class**="": backend k8s storage class - -**--backend-k8s-storage-rwx**: backend k8s storage access mode, should ReadWriteMany (RWX) instead of ReadWriteOnce (RWO) be used? (default: true) - -**--backend-k8s-volume-size**="": backend k8s volume size (default 10G) (default: 10G) - -**--backend-local-temp-dir**="": set a different temp dir to clone workflows into (default: /var/folders/nr/x23mhfm55616f3w8xd0lwmdh0000gn/T/) - -**--backend-no-proxy**="": if set, pass the environment variable down as "NO_PROXY" to steps - -**--commit-author-avatar**="": - -**--commit-author-email**="": - -**--commit-author-name**="": - -**--commit-branch**="": - -**--commit-message**="": - -**--commit-ref**="": - -**--commit-refspec**="": - -**--commit-sha**="": - -**--connect-retry-count**="": number of times to retry connecting to the server (default: 0) - -**--connect-retry-delay**="": duration to wait before retrying to connect to the server (default: 0s) - -**--env**="": - -**--forge-type**="": - -**--forge-url**="": - -**--local**: run from local directory - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--netrc-machine**="": - -**--netrc-password**="": - -**--netrc-username**="": - -**--network**="": external networks - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pipeline-created**="": (default: 0) - -**--pipeline-event**="": (default: manual) - -**--pipeline-finished**="": (default: 0) - -**--pipeline-number**="": (default: 0) - -**--pipeline-parent**="": (default: 0) - -**--pipeline-started**="": (default: 0) - -**--pipeline-status**="": - -**--pipeline-target**="": - -**--pipeline-url**="": - -**--pretty**: enable pretty-printed debug output - -**--prev-commit-author-avatar**="": - -**--prev-commit-author-email**="": - -**--prev-commit-author-name**="": - -**--prev-commit-branch**="": - -**--prev-commit-message**="": - -**--prev-commit-ref**="": - -**--prev-commit-refspec**="": - -**--prev-commit-sha**="": - -**--prev-pipeline-created**="": (default: 0) - -**--prev-pipeline-event**="": - -**--prev-pipeline-finished**="": (default: 0) - -**--prev-pipeline-number**="": (default: 0) - -**--prev-pipeline-started**="": (default: 0) - -**--prev-pipeline-status**="": - -**--prev-pipeline-url**="": - -**--privileged**="": privileged plugins (default: "plugins/docker", "plugins/gcr", "plugins/ecr", "woodpeckerci/plugin-docker-buildx", "codeberg.org/woodpecker-plugins/docker-buildx") - -**--repo**="": full repo name - -**--repo-clone-ssh-url**="": - -**--repo-clone-url**="": - -**--repo-private**="": - -**--repo-remote-id**="": - -**--repo-trusted**: - -**--repo-url**="": - -**--server, -s**="": server address - -**--step-name**="": (default: 0) - -**--system-name**="": (default: woodpecker) - -**--system-platform**="": - -**--system-url**="": (default: https://github.com/woodpecker-ci/woodpecker) - -**--timeout**="": pipeline timeout (default: 0s) - -**--token, -t**="": server auth token - -**--volumes**="": pipeline volumes - -**--workflow-name**="": (default: 0) - -**--workflow-number**="": (default: 0) - -**--workspace-base**="": (default: /woodpecker) - -**--workspace-path**="": (default: src) - -## info - -show information about the current user - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -## registry - -manage registries - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### add - -adds a registry - -**--hostname**="": registry hostname (default: docker.io) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--password**="": registry password - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -**--username**="": registry username - -### rm - -remove a registry - -**--hostname**="": registry hostname (default: docker.io) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### update - -update a registry - -**--hostname**="": registry hostname (default: docker.io) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--password**="": registry password - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -**--username**="": registry username - -### info - -display registry info - -**--hostname**="": registry hostname (default: docker.io) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### ls - -list registries - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -## secret - -manage secrets - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### add - -adds a secret - -**--event**="": secret limited to these events - -**--global**: global secret - -**--image**="": secret limited to these images - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--name**="": secret name - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -**--value**="": secret value - -### rm - -remove a secret - -**--global**: global secret - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--name**="": secret name - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### update - -update a secret - -**--event**="": secret limited to these events - -**--global**: global secret - -**--image**="": secret limited to these images - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--name**="": secret name - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -**--value**="": secret value - -### info - -display secret info - -**--global**: global secret - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--name**="": secret name - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### ls - -list secrets - -**--global**: global secret - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -## repo - -manage repositories - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### ls - -list all repos - -**--format**="": format output (default: {{ .FullName }} (id: {{ .ID }}, forgeRemoteID: {{ .ForgeRemoteID }})) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--org**="": filter by organization - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### info - -show repository details - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### add - -add a repository - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### update - -update a repository - -**--config**="": repository configuration path (e.g. .woodpecker.yml) - -**--gated**: repository is gated - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pipeline-counter**="": repository starting pipeline number (default: 0) - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--timeout**="": repository timeout (default: 0s) - -**--token, -t**="": server auth token - -**--trusted**: repository is trusted - -**--unsafe**: validate updating the pipeline-counter is unsafe - -**--visibility**="": repository visibility - -### rm - -remove a repository - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### repair - -repair repository webhooks - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### chown - -assume ownership of a repository - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### sync - -synchronize the repository list - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -## user - -manage users - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### ls - -list all users - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### info - -show user details - -**--format**="": format output (default: {{ .Login }}) - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### add - -adds a user - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### rm - -remove a user - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -## lint - -lint a pipeline configuration file - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -## log-level - -get the logging level of the server, or set it with [level] - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -## cron - -manage cron jobs - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### add - -add a cron job - -**--branch**="": cron branch - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--name**="": cron name - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--schedule**="": cron schedule - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### rm - -remove a cron job - -**--id**="": cron id - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### update - -update a cron job - -**--branch**="": cron branch - -**--id**="": cron id - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--name**="": cron name - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--schedule**="": cron schedule - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### info - -display info about a cron job - -**--id**="": cron id - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token - -### ls - -list cron jobs - -**--log-file**="": where logs are written to. 'stdout' and 'stderr' can be used as special keywords (default: stderr) - -**--log-level**="": set logging level (default: info) - -**--nocolor**: disable colored debug output, only has effect if pretty output is set too - -**--pretty**: enable pretty-printed debug output - -**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) - -**--server, -s**="": server address - -**--token, -t**="": server auth token diff --git a/docs/versioned_docs/version-2.3/10-intro.md b/docs/versioned_docs/version-2.6/10-intro.md similarity index 96% rename from docs/versioned_docs/version-2.3/10-intro.md rename to docs/versioned_docs/version-2.6/10-intro.md index 276dcb000..309c6f1af 100644 --- a/docs/versioned_docs/version-2.3/10-intro.md +++ b/docs/versioned_docs/version-2.6/10-intro.md @@ -75,13 +75,13 @@ kubectl apply -f $PLUGIN_TEMPLATE ```yaml title=".woodpecker.yaml" steps: - deploy-to-k8s: + - name: deploy-to-k8s image: laszlocloud/my-k8s-plugin settings: template: config/k8s/service.yaml ``` -See [plugin docs](./20-usage/51-plugins/10-overview.md). +See [plugin docs](./20-usage/51-plugins/51-overview.md). ## Continue reading diff --git a/docs/versioned_docs/version-2.3/20-usage/10-intro.md b/docs/versioned_docs/version-2.6/20-usage/10-intro.md similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/10-intro.md rename to docs/versioned_docs/version-2.6/20-usage/10-intro.md diff --git a/docs/versioned_docs/version-2.6/20-usage/100-troubleshooting.md b/docs/versioned_docs/version-2.6/20-usage/100-troubleshooting.md new file mode 100644 index 000000000..b961530f4 --- /dev/null +++ b/docs/versioned_docs/version-2.6/20-usage/100-troubleshooting.md @@ -0,0 +1,37 @@ +# Troubleshooting + +## How to debug clone issues + +(And what to do with an error message like `fatal: could not read Username for 'https://': No such device or address`) + +This error can have multiple causes. If you use internal repositories you might have to enable `WOODPECKER_AUTHENTICATE_PUBLIC_REPOS`: + +```ini +WOODPECKER_AUTHENTICATE_PUBLIC_REPOS=true +``` + +If that does not work, try to make sure the container can reach your git server. In order to do that disable git checkout and make the container "hang": + +```yaml +skip_clone: true + +steps: + build: + image: debian:stable-backports + commands: + - apt update + - apt install -y inetutils-ping wget + - ping -c 4 git.example.com + - wget git.example.com + - sleep 9999999 +``` + +Get the container id using `docker ps` and copy the id from the first column. Enter the container with: `docker exec -it 1234asdf bash` (replace `1234asdf` with the docker id). Then try to clone the git repository with the commands from the failing pipeline: + +```bash +git init +git remote add origin https://git.example.com/username/repo.git +git fetch --no-tags origin +refs/heads/branch: +``` + +(replace the url AND the branch with the correct values, use your username and password as log in values) diff --git a/docs/versioned_docs/version-2.3/20-usage/15-terminiology/architecture.excalidraw b/docs/versioned_docs/version-2.6/20-usage/15-terminology/architecture.excalidraw similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/15-terminiology/architecture.excalidraw rename to docs/versioned_docs/version-2.6/20-usage/15-terminology/architecture.excalidraw diff --git a/docs/versioned_docs/version-2.3/20-usage/15-terminiology/architecture.svg b/docs/versioned_docs/version-2.6/20-usage/15-terminology/architecture.svg similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/15-terminiology/architecture.svg rename to docs/versioned_docs/version-2.6/20-usage/15-terminology/architecture.svg diff --git a/docs/versioned_docs/version-2.3/20-usage/15-terminiology/index.md b/docs/versioned_docs/version-2.6/20-usage/15-terminology/index.md similarity index 92% rename from docs/versioned_docs/version-2.3/20-usage/15-terminiology/index.md rename to docs/versioned_docs/version-2.6/20-usage/15-terminology/index.md index 4e4f61489..5e9d8e5de 100644 --- a/docs/versioned_docs/version-2.3/20-usage/15-terminiology/index.md +++ b/docs/versioned_docs/version-2.6/20-usage/15-terminology/index.md @@ -31,6 +31,7 @@ - **YAML File**: A file format used to define and configure [workflows][Workflow]. - **Dependency**: [Workflows][Workflow] can depend on each other, and if possible, they are executed in parallel. - **Status**: Status refers to the outcome of a step or [workflow][Workflow] after it has been executed, determined by the internal command exit code. At the end of a [workflow][Workflow], its status is sent to the [forge][Forge]. +- **Service extension**: Some parts of Woodpecker internal services like secrets storage or config fetcher can be replaced through service extensions. ## Pipeline events @@ -49,13 +50,14 @@ Sometimes there are multiple terms that can be used to describe something. This - Environment variables `*_LINK` should be called `*_URL`. In the code use `URL()` instead of `Link()` - Use the term **pipelines** instead of the previous **builds** - Use the term **steps** instead of the previous **jobs** +- Use the prefix `WOODPECKER_EXPERT_` for advanced environment variables that are normally not required to be set by users [Pipeline]: ../20-workflow-syntax.md [Workflow]: ../25-workflows.md -[Forge]: ../../30-administration/11-forges/10-overview.md -[Plugin]: ../51-plugins/10-overview.md +[Forge]: ../../30-administration/11-forges/11-overview.md +[Plugin]: ../51-plugins/51-overview.md [Workspace]: ../20-workflow-syntax.md#workspace [Matrix]: ../30-matrix-workflows.md [Docker]: ../../30-administration/22-backends/10-docker.md diff --git a/docs/versioned_docs/version-2.3/20-usage/15-terminiology/pipeline-workflow-step.excalidraw b/docs/versioned_docs/version-2.6/20-usage/15-terminology/pipeline-workflow-step.excalidraw similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/15-terminiology/pipeline-workflow-step.excalidraw rename to docs/versioned_docs/version-2.6/20-usage/15-terminology/pipeline-workflow-step.excalidraw diff --git a/docs/versioned_docs/version-2.3/20-usage/15-terminiology/pipeline-workflow-step.svg b/docs/versioned_docs/version-2.6/20-usage/15-terminology/pipeline-workflow-step.svg similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/15-terminiology/pipeline-workflow-step.svg rename to docs/versioned_docs/version-2.6/20-usage/15-terminology/pipeline-workflow-step.svg diff --git a/docs/versioned_docs/version-2.3/20-usage/20-workflow-syntax.md b/docs/versioned_docs/version-2.6/20-usage/20-workflow-syntax.md similarity index 79% rename from docs/versioned_docs/version-2.3/20-usage/20-workflow-syntax.md rename to docs/versioned_docs/version-2.6/20-usage/20-workflow-syntax.md index 1f9d80248..7b966fc48 100644 --- a/docs/versioned_docs/version-2.3/20-usage/20-workflow-syntax.md +++ b/docs/versioned_docs/version-2.6/20-usage/20-workflow-syntax.md @@ -1,6 +1,10 @@ # Workflow syntax -The workflow section defines a list of steps to build, test and deploy your code. Steps are executed serially, in the order in which they are defined. If a step returns a non-zero exit code, the workflow and therefore all other workflows and the pipeline immediately aborts and returns a failure status. +The Workflow section defines a list of steps to build, test and deploy your code. The steps are executed serially in the order in which they are defined. If a step returns a non-zero exit code, the workflow and therefore the entire pipeline terminates immediately and returns an error status. + +:::note +An exception to this rule are steps with a [`status: [failure]`](#status) condition, which ensures that they are executed in the case of a failed run. +::: Example steps: @@ -50,7 +54,8 @@ git commit -m "updated README [CI SKIP]" ## Steps -Every step of your workflow executes commands inside a specified container. The defined commands are executed serially. +Every step of your workflow executes commands inside a specified container.
+The defined steps are executed in sequence by default, if they should run in parallel you can use [`depends_on`](./20-workflow-syntax.md#depends_on).
The associated commit is checked out with git to a workspace which is mounted to every step of the workflow as the working directory. ```diff @@ -160,17 +165,20 @@ Only build steps can define commands. You cannot use commands with plugins or se Allows you to specify the entrypoint for containers. Note that this must be a list of the command and its arguments (e.g. `["/bin/sh", "-c"]`). +If you define [`commands`](#commands), the default entrypoint will be `["/bin/sh", "-c", "echo $CI_SCRIPT | base64 -d | /bin/sh -e"]`. +You can also use a custom shell with `CI_SCRIPT` (Base64-encoded) if you set `commands`. + ### `environment` Woodpecker provides the ability to pass environment variables to individual steps. -For more details check the [environment docs](./50-environment.md). +For more details, check the [environment docs](./50-environment.md). ### `secrets` Woodpecker provides the ability to store named parameters external to the YAML configuration file, in a central secret store. These secrets can be passed to individual steps of the workflow at runtime. -For more details check the [secrets docs](./40-secrets.md). +For more details, check the [secrets docs](./40-secrets.md). ### `failure` @@ -188,7 +196,8 @@ Some of the steps may be allowed to fail without causing the whole workflow and ### `when` - Conditional Execution -Woodpecker supports defining a list of conditions for a step by using a `when` block. If at least one of the conditions in the `when` block evaluate to true the step is executed, otherwise it is skipped. A condition can be a check like: +Woodpecker supports defining a list of conditions for a step by using a `when` block. If at least one of the conditions in the `when` block evaluate to true the step is executed, otherwise it is skipped. A condition is evaluated to true if _all_ subconditions are true. +A condition can be a check like: ```diff steps: @@ -203,6 +212,11 @@ Woodpecker supports defining a list of conditions for a step by using a `when` b + branch: main ``` +The `slack` step is executed if one of these conditions is met: + +1. The pipeline is executed from a pull request in the repo `test/test` +2. The pipeline is executed from a push to `maiǹ` + #### `repo` Example conditional execution by repository: @@ -352,16 +366,6 @@ when: - platform: [linux/*, windows/amd64] ``` -#### `environment` - -Execute a step for deployment events matching the target deployment environment: - -```yaml -when: - - environment: production - - event: deployment -``` - #### `matrix` Execute a step for a single matrix permutation: @@ -398,16 +402,19 @@ when: You can use [glob patterns](https://github.com/bmatcuk/doublestar#patterns) to match the changed files and specify if the step should run if a file matching that pattern has been changed `include` or if some files have **not** been changed `exclude`. +For pipelines without file changes (empty commits or on events without file changes like `tag`), you can use `on_empty` to set whether this condition should be **true** _(default)_ or **false** in these cases. + ```yaml when: - path: include: ['.woodpecker/*.yaml', '*.ini'] exclude: ['*.md', 'docs/**'] ignore_message: '[ALL]' + on_empty: true ``` :::info -Passing a defined ignore-message like `[ALL]` inside the commit message will ignore all path conditions. +Passing a defined ignore-message like `[ALL]` inside the commit message will ignore all path conditions and the `on_empty` setting. ::: #### `evaluate` @@ -474,6 +481,19 @@ Normally steps of a workflow are executed serially in the order in which they ar - go test ``` +:::note +You can define a step to start immediately without dependencies by adding an empty `depends_on: []`. By setting `depends_on` on a single step all other steps will be immediately executed as well if no further dependencies are specified. + +```yaml +steps: + - name: check code format + image: mstruebing/editorconfig-checker + depends_on: [] # enable parallel steps + ... +``` + +::: + ### `volumes` Woodpecker gives the ability to define Docker volumes in the YAML. You can use this parameter to mount files or folders on the host machine into your containers. @@ -556,8 +576,12 @@ git clone https://github.com/octocat/hello-world \ /go/src/github.com/octocat/hello-world ``` + + ## `matrix` + + Woodpecker has integrated support for matrix builds. Woodpecker executes a separate build task for each combination in the matrix, allowing you to build and test a single commit against multiple configurations. For more details check the [matrix build docs](./30-matrix-workflows.md). @@ -566,10 +590,10 @@ For more details check the [matrix build docs](./30-matrix-workflows.md). You can set labels for your workflow to select an agent to execute the workflow on. An agent will pick up and run a workflow when **every** label assigned to it matches the agents labels. -To set additional agent labels check the [agent configuration options](../30-administration/15-agent-config.md#woodpecker_filter_labels). Agents will have at least four default labels: `platform=agent-os/agent-arch`, `hostname=my-agent`, `backend=docker` (type of the agent backend) and `repo=*`. Agents can use a `*` as a wildcard for a label. For example `repo=*` will match every repo. +To set additional agent labels, check the [agent configuration options](../30-administration/15-agent-config.md#woodpecker_filter_labels). Agents will have at least four default labels: `platform=agent-os/agent-arch`, `hostname=my-agent`, `backend=docker` (type of the agent backend) and `repo=*`. Agents can use a `*` as a wildcard for a label. For example `repo=*` will match every repo. Workflow labels with an empty value will be ignored. -By default each workflow has at least the `repo=your-user/your-repo-name` label. If you have set the [platform attribute](#platform) for your workflow it will have a label like `platform=your-os/your-arch` as well. +By default, each workflow has at least the `repo=your-user/your-repo-name` label. If you have set the [platform attribute](#platform) for your workflow it will have a label like `platform=your-os/your-arch` as well. You can add additional labels as a key value map: @@ -644,7 +668,7 @@ Example configuration to use a custom clone plugin: ```diff clone: - git: + - name: git + image: octocat/custom-git-plugin ``` @@ -694,28 +718,9 @@ skip_clone: true ## `when` - Global workflow conditions -Woodpecker gives the ability to skip whole workflows (not just steps #when---conditional-execution-1) based on certain conditions by a `when` block. If all conditions in the `when` block evaluate to true the workflow is executed, otherwise it is skipped, but treated as successful and other workflows depending on it will still continue. +Woodpecker gives the ability to skip whole workflows ([not just steps](#when---conditional-execution)) based on certain conditions by a `when` block. If all conditions in the `when` block evaluate to true the workflow is executed, otherwise it is skipped, but treated as successful and other workflows depending on it will still continue. -### `repo` - -Example conditional execution by repository: - -```diff -+when: -+ repo: test/test -+ - steps: - - name: slack - image: plugins/slack - settings: - channel: dev -``` - -### `branch` - -:::note -Branch conditions are not applied to tags. -::: +For more information about the specific filters, take a look at the [step-specific `when` filters](#when---conditional-execution). Example conditional execution by branch: @@ -730,126 +735,14 @@ Example conditional execution by branch: channel: dev ``` -The step now triggers on `main`, but also if the target branch of a pull request is `main`. Add an event condition to limit it further to pushes on main only. +The workflow now triggers on `main`, but also if the target branch of a pull request is `main`. -Execute a step if the branch is `main` or `develop`: - -```yaml -when: - branch: [main, develop] -``` - -Execute a step if the branch starts with `prefix/*`: - -```yaml -when: - branch: prefix/* -``` - -Execute a step using custom include and exclude logic: - -```yaml -when: - branch: - include: [main, release/*] - exclude: [release/1.0.0, release/1.1.*] -``` - -### `event` - -:::warning -Some events like the release event will be triggered for multiple actions like: releases, pre-releases and drafts. If you want to apply further filters checkout the [evaluate](#evaluate) filter and the available [environment variables](./50-environment.md#built-in-environment-variables). -::: - -Execute a step if the build event is a `tag`: - -```yaml -when: - event: tag -``` - -Execute a step if the pipeline event is a `push` to a specified branch: - -```diff - when: - event: push -+ branch: main -``` - -Execute a step for all non-pull request events: - -```yaml -when: - event: [push, tag, deployment] -``` - -Execute a step for all build events: - -```yaml -when: - event: [push, pull_request, pull_request_closed, tag, deployment, release] -``` - -### `ref` - -The `ref` filter compares the git reference against which the pipeline is executed. -This allows you to filter, for example, tags that must start with **v**: - -```yaml -when: - event: tag - ref: refs/tags/v* -``` - -### `environment` - -Execute a step for deployment events matching the target deployment environment: - -```yaml -when: - environment: production - event: deployment -``` - -### `instance` - -Execute a step only on a certain Woodpecker instance matching the specified hostname: - -```yaml -when: - instance: stage.woodpecker.company.com -``` - -### `path` - -:::info -Path conditions are applied only to **push** and **pull_request** events. -It is currently **only available** for GitHub, GitLab and Gitea (version 1.18.0 and newer) -::: - -Execute a step only on a pipeline with certain files being changed: - -```yaml -when: - path: 'src/*' -``` - -You can use [glob patterns](https://github.com/bmatcuk/doublestar#patterns) to match the changed files and specify if the step should run if a file matching that pattern has been changed `include` or if some files have **not** been changed `exclude`. - -```yaml -when: - path: - include: ['.woodpecker/*.yaml', '*.ini'] - exclude: ['*.md', 'docs/**'] - ignore_message: '[ALL]' -``` - -:::info -Passing a defined ignore-message like `[ALL]` inside the commit message will ignore all path conditions. -::: + ## `depends_on` + + Woodpecker supports to define multiple workflows for a repository. Those workflows will run independent from each other. To depend them on each other you can use the [`depends_on`](./25-workflows.md#flow-control) keyword. ## `runs_on` @@ -861,7 +754,7 @@ Workflows that should run even on failure should set the `runs_on` tag. See [her Woodpecker gives the ability to configure privileged mode in the YAML. You can use this parameter to launch containers with escalated capabilities. :::info -Privileged mode is only available to trusted repositories and for security reasons should only be used in private environments. See [project settings](./71-project-settings.md#trusted) to enable trusted mode. +Privileged mode is only available to trusted repositories and for security reasons should only be used in private environments. See [project settings](./75-project-settings.md#trusted) to enable trusted mode. ::: ```diff diff --git a/docs/versioned_docs/version-2.3/20-usage/25-workflows.md b/docs/versioned_docs/version-2.6/20-usage/25-workflows.md similarity index 96% rename from docs/versioned_docs/version-2.3/20-usage/25-workflows.md rename to docs/versioned_docs/version-2.6/20-usage/25-workflows.md index 1a3f40fc8..5adc39f85 100644 --- a/docs/versioned_docs/version-2.3/20-usage/25-workflows.md +++ b/docs/versioned_docs/version-2.6/20-usage/25-workflows.md @@ -6,7 +6,7 @@ In case there is a single configuration in `.woodpecker.yaml` Woodpecker will cr By placing the configurations in a folder which is by default named `.woodpecker/` Woodpecker will create a pipeline with multiple workflows each named by the file they are defined in. Only `.yml` and `.yaml` files will be used and files in any subfolders like `.woodpecker/sub-folder/test.yaml` will be ignored. -You can also set some custom path like `.my-ci/pipelines/` instead of `.woodpecker/` in the [project settings](./71-project-settings.md). +You can also set some custom path like `.my-ci/pipelines/` instead of `.woodpecker/` in the [project settings](./75-project-settings.md). ## Benefits of using workflows @@ -18,7 +18,7 @@ You can also set some custom path like `.my-ci/pipelines/` instead of `.woodpeck :::warning Please note that files are only shared between steps of the same workflow (see [File changes are incremental](./20-workflow-syntax.md#file-changes-are-incremental)). That means you cannot access artifacts e.g. from the `build` workflow in the `deploy` workflow. -If you still need to pass artifacts between the workflows you need use some storage [plugin](./51-plugins/10-overview.md) (e.g. one which stores files in an Amazon S3 bucket). +If you still need to pass artifacts between the workflows you need use some storage [plugin](./51-plugins/51-overview.md) (e.g. one which stores files in an Amazon S3 bucket). ::: ```bash diff --git a/docs/versioned_docs/version-2.3/20-usage/30-matrix-workflows.md b/docs/versioned_docs/version-2.6/20-usage/30-matrix-workflows.md similarity index 99% rename from docs/versioned_docs/version-2.3/20-usage/30-matrix-workflows.md rename to docs/versioned_docs/version-2.6/20-usage/30-matrix-workflows.md index 939530e6c..954820523 100644 --- a/docs/versioned_docs/version-2.3/20-usage/30-matrix-workflows.md +++ b/docs/versioned_docs/version-2.6/20-usage/30-matrix-workflows.md @@ -139,5 +139,5 @@ steps: ``` :::note -If you want to control the architecture of a pipeline on a Kubernetes runner, see [the nodeSelector documentation of the Kubernetes backend](../30-administration/22-backends/40-kubernetes.md#nodeselector). +If you want to control the architecture of a pipeline on a Kubernetes runner, see [the nodeSelector documentation of the Kubernetes backend](../30-administration/22-backends/40-kubernetes.md#node-selector). ::: diff --git a/docs/versioned_docs/version-2.3/20-usage/40-secrets.md b/docs/versioned_docs/version-2.6/20-usage/40-secrets.md similarity index 75% rename from docs/versioned_docs/version-2.3/20-usage/40-secrets.md rename to docs/versioned_docs/version-2.6/20-usage/40-secrets.md index 09fc00c45..1b55d9ce1 100644 --- a/docs/versioned_docs/version-2.3/20-usage/40-secrets.md +++ b/docs/versioned_docs/version-2.6/20-usage/40-secrets.md @@ -21,23 +21,27 @@ once their usage is declared in the `secrets` section: - name: docker image: docker commands: -+ - echo $DOCKER_USERNAME ++ - echo $docker_username + - echo $DOCKER_PASSWORD -+ secrets: [ docker_username, docker_password ] ++ secrets: [ docker_username, DOCKER_PASSWORD ] ``` -### Use secrets in settings +The case of the environment variables is not changed, but secret matching is done case-insensitively. In the example above, `DOCKER_PASSWORD` would also match if the secret is called `docker_password`. -Alternatively, you can get a `setting` from secrets using the `from_secret` syntax. -In this example, the secret named `secret_token` would be passed to the setting named `token`, which will be available in the plugin as environment variable named `PLUGIN_TOKEN`. See [Plugins](./51-plugins/20-creating-plugins.md#settings) for details. +### Use secrets in settings and environment -**NOTE:** the `from_secret` syntax only works with the newer `settings` block. +You can set an setting or environment value from secrets using the `from_secret` syntax. + +In this example, the secret named `secret_token` would be passed to the setting named `token`,which will be available in the plugin as environment variable named `PLUGIN_TOKEN` (See [plugins](./51-plugins/20-creating-plugins.md#settings) for details), and to the environment variable `TOKEN_ENV`. ```diff steps: - name: docker image: my-plugin - settings: ++ environment: ++ TOKEN_ENV: ++ from_secret: secret_token ++ settings: + token: + from_secret: secret_token ``` @@ -51,33 +55,20 @@ Please note parameter expressions are subject to pre-processing. When using secr - name: docker image: docker commands: -- - echo ${DOCKER_USERNAME} +- - echo ${docker_username} - - echo ${DOCKER_PASSWORD} -+ - echo $${DOCKER_USERNAME} ++ - echo $${docker_username} + - echo $${DOCKER_PASSWORD} - secrets: [ docker_username, docker_password ] -``` - -### Alternate Names - -There may be scenarios where you are required to store secrets using alternate names. You can map the alternate secret name to the expected name using the below syntax: - -```diff - steps: - - name: docker - image: plugins/docker - repo: octocat/hello-world - tags: latest -+ secrets: -+ - source: docker_prod_password -+ target: docker_password + secrets: [ docker_username, DOCKER_PASSWORD ] ``` ### Use in Pull Requests events Secrets are not exposed to pull requests by default. You can override this behavior by creating the secret and enabling the `pull_request` event type, either in UI or by CLI, see below. -**NOTE:** Please be careful when exposing secrets to pull requests. If your repository is open source and accepts pull requests your secrets are not safe. A bad actor can submit a malicious pull request that exposes your secrets. +:::note +Please be careful when exposing secrets to pull requests. If your repository is open source and accepts pull requests your secrets are not safe. A bad actor can submit a malicious pull request that exposes your secrets. +::: ## Image filter diff --git a/docs/versioned_docs/version-2.3/20-usage/41-registries.md b/docs/versioned_docs/version-2.6/20-usage/41-registries.md similarity index 94% rename from docs/versioned_docs/version-2.3/20-usage/41-registries.md rename to docs/versioned_docs/version-2.6/20-usage/41-registries.md index b87ae55c4..8508da876 100644 --- a/docs/versioned_docs/version-2.3/20-usage/41-registries.md +++ b/docs/versioned_docs/version-2.6/20-usage/41-registries.md @@ -35,6 +35,10 @@ Example registry hostname matching logic: - Hostname `docker.io` matches `bradyrydzewski/golang` - Hostname `docker.io` matches `bradyrydzewski/golang:latest` +:::note +The flow above doesn't work in Kubernetes. There is [workaround](../30-administration/22-backends/40-kubernetes.md#images-from-private-registries). +::: + ## Global registry support To make a private registry globally available, check the [server configuration docs](../30-administration/10-server-config.md#global-registry-setting). diff --git a/docs/versioned_docs/version-2.3/20-usage/45-cron.md b/docs/versioned_docs/version-2.6/20-usage/45-cron.md similarity index 88% rename from docs/versioned_docs/version-2.3/20-usage/45-cron.md rename to docs/versioned_docs/version-2.6/20-usage/45-cron.md index 994e022bc..95ee8202e 100644 --- a/docs/versioned_docs/version-2.3/20-usage/45-cron.md +++ b/docs/versioned_docs/version-2.6/20-usage/45-cron.md @@ -19,11 +19,11 @@ To configure cron jobs you need at least push access to the repository. + cron: "name of the cron job" # if you only want to execute this step by a specific cron job ``` -1. Create a new cron job in the repository settings: +2. Create a new cron job in the repository settings: ![cron settings](./cron-settings.png) - The supported schedule syntax can be found at . If you need general understanding of the cron syntax is a good place to start and experiment. + The supported schedule syntax can be found at . If you need general understanding of the cron syntax is a good place to start and experiment. Examples: `@every 5m`, `@daily`, `0 30 * * * *` ... diff --git a/docs/versioned_docs/version-2.3/20-usage/50-environment.md b/docs/versioned_docs/version-2.6/20-usage/50-environment.md similarity index 96% rename from docs/versioned_docs/version-2.3/20-usage/50-environment.md rename to docs/versioned_docs/version-2.6/20-usage/50-environment.md index 5e45ba0d4..299bb8f53 100644 --- a/docs/versioned_docs/version-2.3/20-usage/50-environment.md +++ b/docs/versioned_docs/version-2.6/20-usage/50-environment.md @@ -7,9 +7,9 @@ Woodpecker provides the ability to pass environment variables to individual pipe - name: build image: golang + environment: -+ - CGO=0 -+ - GOOS=linux -+ - GOARCH=amd64 ++ CGO: 0 ++ GOOS: linux ++ GOARCH: amd64 commands: - go build - go test @@ -81,10 +81,11 @@ This is the reference list of all environment variables available to your pipeli | | **Current pipeline** | | `CI_PIPELINE_NUMBER` | pipeline number | | `CI_PIPELINE_PARENT` | number of parent pipeline | -| `CI_PIPELINE_EVENT` | pipeline event (see [pipeline events](../20-usage/15-terminiology/index.md#pipeline-events)) | +| `CI_PIPELINE_EVENT` | pipeline event (see [pipeline events](../20-usage/15-terminology/index.md#pipeline-events)) | | `CI_PIPELINE_URL` | link to the web UI for the pipeline | | `CI_PIPELINE_FORGE_URL` | link to the forge's web UI for the commit(s) or tag that triggered the pipeline | | `CI_PIPELINE_DEPLOY_TARGET` | pipeline deploy target for `deployment` events (i.e. production) | +| `CI_PIPELINE_DEPLOY_TASK` | pipeline deploy task for `deployment` events (i.e. migration) | | `CI_PIPELINE_STATUS` | pipeline status (success, failure) | | `CI_PIPELINE_CREATED` | pipeline created UNIX timestamp | | `CI_PIPELINE_STARTED` | pipeline started UNIX timestamp | @@ -114,10 +115,11 @@ This is the reference list of all environment variables available to your pipeli | | **Previous pipeline** | | `CI_PREV_PIPELINE_NUMBER` | previous pipeline number | | `CI_PREV_PIPELINE_PARENT` | previous pipeline number of parent pipeline | -| `CI_PREV_PIPELINE_EVENT` | previous pipeline event (see [pipeline events](../20-usage/15-terminiology/index.md#pipeline-events)) | +| `CI_PREV_PIPELINE_EVENT` | previous pipeline event (see [pipeline events](../20-usage/15-terminology/index.md#pipeline-events)) | | `CI_PREV_PIPELINE_URL` | previous pipeline link in CI | | `CI_PREV_PIPELINE_FORGE_URL` | previous pipeline link to event in forge | | `CI_PREV_PIPELINE_DEPLOY_TARGET` | previous pipeline deploy target for `deployment` events (ie production) | +| `CI_PREV_PIPELINE_DEPLOY_TASK` | previous pipeline deploy task for `deployment` events (ie migration) | | `CI_PREV_PIPELINE_STATUS` | previous pipeline status (success, failure) | | `CI_PREV_PIPELINE_CREATED` | previous pipeline created UNIX timestamp | | `CI_PREV_PIPELINE_STARTED` | previous pipeline started UNIX timestamp | diff --git a/docs/versioned_docs/version-2.3/20-usage/51-plugins/20-creating-plugins.md b/docs/versioned_docs/version-2.6/20-usage/51-plugins/20-creating-plugins.md similarity index 81% rename from docs/versioned_docs/version-2.3/20-usage/51-plugins/20-creating-plugins.md rename to docs/versioned_docs/version-2.6/20-usage/51-plugins/20-creating-plugins.md index adbb4df3e..8a0ea5920 100644 --- a/docs/versioned_docs/version-2.3/20-usage/51-plugins/20-creating-plugins.md +++ b/docs/versioned_docs/version-2.6/20-usage/51-plugins/20-creating-plugins.md @@ -42,12 +42,29 @@ Values like this are converted to JSON and then passed to your plugin. In the ex ### Secrets -Secrets should be passed as settings too. Therefore, users should use [`from_secret`](../40-secrets.md#use-secrets-in-settings). +Secrets should be passed as settings too. Therefore, users should use [`from_secret`](../40-secrets.md#use-secrets-in-settings-and-environment). ## Plugin library For Go, we provide a plugin library you can use to get easy access to internal env vars and your settings. See . +## Metadata + +In your documentation, you can use a Markdown header to define metadata for your plugin. This data is used by [our plugin index](/plugins). + +Supported metadata: + +- `name`: The plugin's full name +- `icon`: URL to your plugin's icon +- `description`: A short description of what it's doing +- `author`: Your name +- `tags`: List of keywords (e.g. `[git, clone]` for the clone plugin) +- `containerImage`: name of the container image +- `containerImageUrl`: link to the container image +- `url`: homepage or repository of your plugin + +If you want your plugin to be listed in the index, you should add as many fields as possible, but only `name` is required. + ## Example plugin This provides a brief tutorial for creating a Woodpecker webhook plugin, using simple shell scripting, to make HTTP requests during the build pipeline. @@ -118,5 +135,5 @@ docker run --rm \ These should also be built for different OS/architectures. - Use [built-in env vars](../50-environment.md#built-in-environment-variables) where possible. - Do not use any configuration except settings (and internal env vars). This means: Don't require using [`environment`](../50-environment.md) and don't require specific secret names. -- Add a `docs.md` file, listing all your settings and plugin metadata ([example](https://codeberg.org/woodpecker-plugins/plugin-docker-buildx/src/branch/main/docs.md)). -- Add your plugin to the [plugin index](/plugins) using your `docs.md` ([the example above in the index](https://woodpecker-ci.org/plugins/Docker%20Buildx)). +- Add a `docs.md` file, listing all your settings and plugin metadata ([example](https://github.com/woodpecker-ci/plugin-git/blob/main/docs.md)). +- Add your plugin to the [plugin index](/plugins) using your `docs.md` ([the example above in the index](https://woodpecker-ci.org/plugins/Git%20Clone)). diff --git a/docs/versioned_docs/version-2.3/20-usage/51-plugins/10-overview.md b/docs/versioned_docs/version-2.6/20-usage/51-plugins/51-overview.md similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/51-plugins/10-overview.md rename to docs/versioned_docs/version-2.6/20-usage/51-plugins/51-overview.md diff --git a/docs/versioned_docs/version-2.3/20-usage/51-plugins/_category_.yaml b/docs/versioned_docs/version-2.6/20-usage/51-plugins/_category_.yaml similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/51-plugins/_category_.yaml rename to docs/versioned_docs/version-2.6/20-usage/51-plugins/_category_.yaml diff --git a/docs/versioned_docs/version-2.3/20-usage/60-services.md b/docs/versioned_docs/version-2.6/20-usage/60-services.md similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/60-services.md rename to docs/versioned_docs/version-2.6/20-usage/60-services.md diff --git a/docs/versioned_docs/version-2.3/20-usage/70-volumes.md b/docs/versioned_docs/version-2.6/20-usage/70-volumes.md similarity index 93% rename from docs/versioned_docs/version-2.3/20-usage/70-volumes.md rename to docs/versioned_docs/version-2.6/20-usage/70-volumes.md index 3397e879c..6897053fb 100644 --- a/docs/versioned_docs/version-2.3/20-usage/70-volumes.md +++ b/docs/versioned_docs/version-2.6/20-usage/70-volumes.md @@ -3,7 +3,7 @@ Woodpecker gives the ability to define Docker volumes in the YAML. You can use this parameter to mount files or folders on the host machine into your containers. :::note -Volumes are only available to trusted repositories and for security reasons should only be used in private environments. See [project settings](./71-project-settings.md#trusted) to enable trusted mode. +Volumes are only available to trusted repositories and for security reasons should only be used in private environments. See [project settings](./75-project-settings.md#trusted) to enable trusted mode. ::: ```diff diff --git a/docs/versioned_docs/version-2.6/20-usage/72-linter.md b/docs/versioned_docs/version-2.6/20-usage/72-linter.md new file mode 100644 index 000000000..4fae3d643 --- /dev/null +++ b/docs/versioned_docs/version-2.6/20-usage/72-linter.md @@ -0,0 +1,62 @@ +# Linter + +Woodpecker automatically lints your workflow files for errors, deprecations and bad habits. Errors and warnings are shown in the UI for any pipelines. + +![errors and warnings in UI](./linter-warnings-errors.png) + +## Running the linter from CLI + +You can run the linter also manually from the CLI: + +```shell +woodpecker-cli lint +``` + +## Bad habit warnings + +Woodpecker warns you if your configuration contains some bad habits. + +### Event filter for all steps + +All your items in `when` blocks should have an `event` filter, so no step runs on all events. This is recommended because if new events are added, your steps probably shouldn't run on those as well. + +Examples of an **incorrect** config for this rule: + +```yaml +when: + - branch: main + - event: tag +``` + +This will trigger the warning because the first item (`branch: main`) does not filter with an event. + +```yaml +steps: + - name: test + when: + branch: main + + - name: deploy + when: + event: tag +``` + +Examples of a **correct** config for this rule: + +```yaml +when: + - branch: main + event: push + - event: tag +``` + +```yaml +steps: + - name: test + when: + event: [tag, push] + + - name: deploy + when: + - event: tag +``` diff --git a/docs/versioned_docs/version-2.3/20-usage/71-project-settings.md b/docs/versioned_docs/version-2.6/20-usage/75-project-settings.md similarity index 86% rename from docs/versioned_docs/version-2.3/20-usage/71-project-settings.md rename to docs/versioned_docs/version-2.6/20-usage/75-project-settings.md index f09402b61..24bdbe605 100644 --- a/docs/versioned_docs/version-2.3/20-usage/71-project-settings.md +++ b/docs/versioned_docs/version-2.6/20-usage/75-project-settings.md @@ -12,18 +12,25 @@ The path to the pipeline config file or folder. By default it is left empty whic Your Version-Control-System will notify Woodpecker about events via webhooks. If you want your pipeline to only run on specific webhooks, you can check them with this setting. -## Project settings - -### Allow pull requests +## Allow pull requests Enables handling webhook's pull request event. If disabled, then pipeline won't run for pull requests. -### Protected +## Allow deployments + +Enables a pipeline to be started with the `deploy` event from a successful pipeline. + +:::danger +Only activate this option if you trust all users who have push access to your repository. +Otherwise, these users will be able to steal secrets that are only available for `deploy` events. +::: + +## Protected Every pipeline initiated by an webhook event needs to be approved by a project members with push permissions before being executed. The protected option can be used as an additional review process before running potentially harmful pipelines. Especially if pipelines can be executed by third-parties through pull-requests. -### Trusted +## Trusted If you set your project to trusted, a pipeline step and by this the underlying containers gets access to escalated capabilities like mounting volumes. @@ -33,7 +40,7 @@ Only server admins can set this option. If you are not a server admin this optio ::: -### Only inject netrc credentials into trusted containers +## Only inject netrc credentials into trusted containers Cloning pipeline step may need git credentials. They are injected via netrc. By default, they're only injected if this option is enabled, the repo is trusted ([see above](#trusted)) or the image is a trusted clone image. If you uncheck the option, git credentials will be injected into any container in clone step. diff --git a/docs/versioned_docs/version-2.3/20-usage/80-badges.md b/docs/versioned_docs/version-2.6/20-usage/80-badges.md similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/80-badges.md rename to docs/versioned_docs/version-2.6/20-usage/80-badges.md diff --git a/docs/versioned_docs/version-2.3/20-usage/90-advanced-usage.md b/docs/versioned_docs/version-2.6/20-usage/90-advanced-usage.md similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/90-advanced-usage.md rename to docs/versioned_docs/version-2.6/20-usage/90-advanced-usage.md diff --git a/docs/versioned_docs/version-2.3/20-usage/_category_.yaml b/docs/versioned_docs/version-2.6/20-usage/_category_.yaml similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/_category_.yaml rename to docs/versioned_docs/version-2.6/20-usage/_category_.yaml diff --git a/docs/versioned_docs/version-2.3/20-usage/cron-settings.png b/docs/versioned_docs/version-2.6/20-usage/cron-settings.png similarity index 100% rename from docs/versioned_docs/version-2.3/20-usage/cron-settings.png rename to docs/versioned_docs/version-2.6/20-usage/cron-settings.png diff --git a/docs/versioned_docs/version-2.6/20-usage/linter-warnings-errors.png b/docs/versioned_docs/version-2.6/20-usage/linter-warnings-errors.png new file mode 100644 index 0000000000000000000000000000000000000000..663e4970495aaa59331d65cdbdbe7c8ee465f9ac GIT binary patch literal 116049 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%0DPrQp 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∾o(r8t=6h~7bq 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 Default: `woodpecker.sqlite` +> Default: `woodpecker.sqlite` if not running inside a container, `/var/lib/woodpecker/woodpecker.sqlite` if running inside a container The database connection string. The default value is the path of the embedded SQLite database file. @@ -441,30 +441,6 @@ WOODPECKER_DATABASE_DATASOURCE=postgres://root:password@1.2.3.4:5432/woodpecker? Read the value for `WOODPECKER_DATABASE_DATASOURCE` from the specified filepath -### `WOODPECKER_ENCRYPTION_KEY` - -> Default: empty - -Encryption key used to encrypt secrets in DB. See [secrets encryption](./40-encryption.md) - -### `WOODPECKER_ENCRYPTION_KEY_FILE` - -> Default: empty - -Read the value for `WOODPECKER_ENCRYPTION_KEY` from the specified filepath - -### `WOODPECKER_ENCRYPTION_TINK_KEYSET_FILE` - -> Default: empty - -Filepath to encryption keyset used to encrypt secrets in DB. See [secrets encryption](./40-encryption.md) - -### `WOODPECKER_ENCRYPTION_DISABLE` - -> Default: empty - -Boolean flag to decrypt secrets in DB and disable server encryption. See [secrets encryption](./40-encryption.md) - ### `WOODPECKER_PROMETHEUS_AUTH_TOKEN` > Default: empty @@ -497,12 +473,6 @@ Supported variables: - `owner`: the repo's owner - `repo`: the repo's name -### `WOODPECKER_ADDONS` - -> Default: empty - -List of addon files. See [addons](./75-addons/00-overview.md). - --- ### `WOODPECKER_LIMIT_MEM_SWAP` @@ -555,6 +525,12 @@ Specify a configuration service endpoint, see [Configuration Extension](./100-ex Specify timeout when fetching the Woodpecker configuration from forge. See for syntax reference. +### `WOODPECKER_FORGE_RETRY` + +> Default: 3 + +Specify how many retries of fetching the Woodpecker configuration from a forge are done before we fail. + ### `WOODPECKER_ENABLE_SWAGGER` > Default: true @@ -567,20 +543,36 @@ Enable the Swagger UI for API documentation. Disable version check in admin web UI. +### `WOODPECKER_LOG_STORE` + +> Default: `database` + +Where to store logs. Possible values: `database` or `file`. + +### `WOODPECKER_LOG_STORE_FILE_PATH` + +> Default empty + +Directory to store logs in if [`WOODPECKER_LOG_STORE`](#woodpecker_log_store) is `file`. + --- ### `WOODPECKER_GITHUB_...` -See [GitHub configuration](forges/github/#configuration) +See [GitHub configuration](./11-forges/20-github.md#configuration) ### `WOODPECKER_GITEA_...` -See [Gitea configuration](forges/gitea/#configuration) +See [Gitea configuration](./11-forges/30-gitea.md#configuration) ### `WOODPECKER_BITBUCKET_...` -See [Bitbucket configuration](forges/bitbucket/#configuration) +See [Bitbucket configuration](./11-forges/50-bitbucket.md#configuration) ### `WOODPECKER_GITLAB_...` -See [Gitlab configuration](forges/gitlab/#configuration) +See [GitLab configuration](./11-forges/40-gitlab.md#configuration) + +### `WOODPECKER_ADDON_FORGE` + +See [addon forges](./11-forges/100-addon.md). diff --git a/docs/versioned_docs/version-2.3/30-administration/100-external-configuration-api.md b/docs/versioned_docs/version-2.6/30-administration/100-external-configuration-api.md similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/100-external-configuration-api.md rename to docs/versioned_docs/version-2.6/30-administration/100-external-configuration-api.md diff --git a/docs/versioned_docs/version-2.6/30-administration/11-forges/100-addon.md b/docs/versioned_docs/version-2.6/30-administration/11-forges/100-addon.md new file mode 100644 index 000000000..e280ed420 --- /dev/null +++ b/docs/versioned_docs/version-2.6/30-administration/11-forges/100-addon.md @@ -0,0 +1,68 @@ +# Addon forges + +If the forge you're using does not comply with [Woodpecker's requirements](../../92-development/02-core-ideas.md#forges) or your setup is too specific to be added to Woodpecker's core, you can write your own forge using an addon forge. + +:::warning +Addon forges are still experimental. Their implementation can change and break at any time. +::: + +:::danger +You need to trust the author of the addon forge you use. It can access authentication codes and other possibly sensitive information. +::: + +## Usage + +To use an addon forge, download the correct addon version. Then, you can add the following to your configuration: + +```ini +WOODPECKER_ADDON_FORGE=/path/to/your/addon/forge/file +``` + +In case you run Woodpecker as container, you probably want to mount the addon binary to `/opt/addons/`. + +### Bug reports + +If you experience bugs, please check which component has the issue. If it's the addon, **do not raise an issue in the main repository**, but rather use the separate addon repositories. To check which component is responsible for the bug, look at the logs. Logs from addons are marked with a special field `addon` containing their addon file name. + +## List of addon forges + +If you wrote or found an addon forge, please add it here so others can find it! + +_Be the first one to add your addon forge!_ + +## Creating addon forges + +Addons use RPC to communicate to the server and are implemented using the [`go-plugin` library](https://github.com/hashicorp/go-plugin). + +### Writing your code + +This example will use the Go language. + +Directly import Woodpecker's Go packages (`go.woodpecker-ci.org/woodpecker/woodpecker/v2`) and use the interfaces and types defined there. + +In the `main` function, just call `"go.woodpecker-ci.org/woodpecker/v2/server/forge/addon".Serve` with a `"go.woodpecker-ci.org/woodpecker/v2/server/forge".Forge` as argument. +This will take care of connecting the addon forge to the server. + +### Example structure + +```go +package main + +import ( + "context" + "net/http" + + "go.woodpecker-ci.org/woodpecker/v2/server/forge/addon" + forgeTypes "go.woodpecker-ci.org/woodpecker/v2/server/forge/types" + "go.woodpecker-ci.org/woodpecker/v2/server/model" +) + +func main() { + addon.Serve(config{}) +} + +type config struct { +} + +// `config` must implement `"go.woodpecker-ci.org/woodpecker/v2/server/forge".Forge`. You must directly use Woodpecker's packages - see imports above. +``` diff --git a/docs/versioned_docs/version-2.6/30-administration/11-forges/11-overview.md b/docs/versioned_docs/version-2.6/30-administration/11-forges/11-overview.md new file mode 100644 index 000000000..ba45adf87 --- /dev/null +++ b/docs/versioned_docs/version-2.6/30-administration/11-forges/11-overview.md @@ -0,0 +1,13 @@ +# Forges + +## Supported features + +| Feature | [GitHub](20-github.md) | [Gitea](30-gitea.md) | [Forgejo](35-forgejo.md) | [Gitlab](40-gitlab.md) | [Bitbucket](50-bitbucket.md) | [Bitbucket Datacenter](60-bitbucket_datacenter.md) | +| ------------------------------------------------------------- | :--------------------: | :------------------: | :----------------------: | :--------------------: | :--------------------------: | :------------------------------------------------: | +| Event: Push | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | +| Event: Tag | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | +| Event: Pull-Request | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | +| Event: Release | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :x: | :x: | +| Event: Deploy | :white_check_mark: | :x: | :x: | :x: | :x: | :x: | +| [Multiple workflows](../../20-usage/25-workflows.md) | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | +| [when.path filter](../../20-usage/20-workflow-syntax.md#path) | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :x: | :x: | diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/20-github.md b/docs/versioned_docs/version-2.6/30-administration/11-forges/20-github.md similarity index 93% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/20-github.md rename to docs/versioned_docs/version-2.6/30-administration/11-forges/20-github.md index f3656f8fd..338e555fd 100644 --- a/docs/versioned_docs/version-2.3/30-administration/11-forges/20-github.md +++ b/docs/versioned_docs/version-2.6/30-administration/11-forges/20-github.md @@ -81,3 +81,9 @@ Read the value for `WOODPECKER_GITHUB_SECRET` from the specified filepath. > Default: `false` Configure if SSL verification should be skipped. + +### `WOODPECKER_GITHUB_PUBLIC_ONLY` + +> Default: `false` + +Configures the GitHub OAuth client to only obtain a token that can manage public repositories. diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/30-gitea.md b/docs/versioned_docs/version-2.6/30-administration/11-forges/30-gitea.md similarity index 84% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/30-gitea.md rename to docs/versioned_docs/version-2.6/30-administration/11-forges/30-gitea.md index b50d2e011..bb8e93c2a 100644 --- a/docs/versioned_docs/version-2.3/30-administration/11-forges/30-gitea.md +++ b/docs/versioned_docs/version-2.6/30-administration/11-forges/30-gitea.md @@ -2,9 +2,9 @@ toc_max_heading_level: 2 --- -# Gitea / Forgejo +# Gitea -Woodpecker comes with built-in support for Gitea and the "soft" fork Forgejo. To enable Gitea you should configure the Woodpecker container using the following environment variables: +Woodpecker comes with built-in support for Gitea. To enable Gitea you should configure the Woodpecker container using the following environment variables: ```ini WOODPECKER_GITEA=true @@ -16,7 +16,7 @@ WOODPECKER_GITEA_SECRET=YOUR_GITEA_CLIENT_SECRET ## Gitea on the same host with containers If you have Gitea also running on the same host within a container, make sure the agent does have access to it. -The agent tries to clone using the URL which Gitea reports through its API. For simplified connectivity, you should add the woodpecker agent to the same docker network as Gitea is in. +The agent tries to clone using the URL which Gitea reports through its API. For simplified connectivity, you should add the Woodpecker agent to the same docker network as Gitea is in. Otherwise, the communication should go via the `docker0` gateway (usually 172.17.0.1). To configure the Docker network if the network's name is `gitea`, configure it like this: @@ -93,3 +93,11 @@ Read the value for `WOODPECKER_GITEA_SECRET` from the specified filepath > Default: `false` Configure if SSL verification should be skipped. + +## Advanced options + +### `WOODPECKER_DEV_GITEA_OAUTH_URL` + +> Default: value of `WOODPECKER_GITEA_URL` + +Configures the user-facing Gitea server address. Should be used if `WOODPECKER_GITEA_URL` points to an internal URL used for API requests. diff --git a/docs/versioned_docs/version-2.6/30-administration/11-forges/35-forgejo.md b/docs/versioned_docs/version-2.6/30-administration/11-forges/35-forgejo.md new file mode 100644 index 000000000..df7793118 --- /dev/null +++ b/docs/versioned_docs/version-2.6/30-administration/11-forges/35-forgejo.md @@ -0,0 +1,97 @@ +--- +toc_max_heading_level: 2 +--- + +# Forgejo + +:::warning +Forgejo support is experimental. +::: + +Woodpecker comes with built-in support for Forgejo. To enable Forgejo you should configure the Woodpecker container using the following environment variables: + +```ini +WOODPECKER_FORGEJO=true +WOODPECKER_FORGEJO_URL=YOUR_FORGEJO_URL +WOODPECKER_FORGEJO_CLIENT=YOUR_FORGEJO_CLIENT +WOODPECKER_FORGEJO_SECRET=YOUR_FORGEJO_CLIENT_SECRET +``` + +## Forgejo on the same host with containers + +If you have Forgejo also running on the same host within a container, make sure the agent does have access to it. +The agent tries to clone using the URL which Forgejo reports through its API. For simplified connectivity, you should add the Woodpecker agent to the same docker network as Forgejo is in. +Otherwise, the communication should go via the `docker0` gateway (usually 172.17.0.1). + +To configure the Docker network if the network's name is `forgejo`, configure it like this: + +```diff title="docker-compose.yaml" + services: + [...] + woodpecker-agent: + [...] + environment: + - [...] ++ - WOODPECKER_BACKEND_DOCKER_NETWORK=forgejo +``` + +## Registration + +Register your application with Forgejo to create your client id and secret. You can find the OAuth applications settings of Forgejo at `https://forgejo./user/settings/`. It is very import the authorization callback URL matches your http(s) scheme and hostname exactly with `https:///authorize` as the path. + +If you run the Woodpecker CI server on the same host as the Forgejo instance, you might also need to allow local connections in Forgejo. Otherwise webhooks will fail. Add the following lines to your Forgejo configuration (usually at `/etc/forgejo/conf/app.ini`). + +```ini +[webhook] +ALLOWED_HOST_LIST=external,loopback +``` + +For reference see [Configuration Cheat Sheet](https://forgejo.org/docs/latest/admin/config-cheat-sheet/#webhook-webhook). + +![forgejo oauth setup](gitea_oauth.gif) + +## Configuration + +This is a full list of configuration options. Please note that many of these options use default configuration values that should work for the majority of installations. + +### `WOODPECKER_FORGEJO` + +> Default: `false` + +Enables the Forgejo driver. + +### `WOODPECKER_FORGEJO_URL` + +> Default: `https://next.forgejo.org` + +Configures the Forgejo server address. + +### `WOODPECKER_FORGEJO_CLIENT` + +> Default: empty + +Configures the Forgejo OAuth client id. This is used to authorize access. + +### `WOODPECKER_FORGEJO_CLIENT_FILE` + +> Default: empty + +Read the value for `WOODPECKER_FORGEJO_CLIENT` from the specified filepath + +### `WOODPECKER_FORGEJO_SECRET` + +> Default: empty + +Configures the Forgejo OAuth client secret. This is used to authorize access. + +### `WOODPECKER_FORGEJO_SECRET_FILE` + +> Default: empty + +Read the value for `WOODPECKER_FORGEJO_SECRET` from the specified filepath + +### `WOODPECKER_FORGEJO_SKIP_VERIFY` + +> Default: `false` + +Configure if SSL verification should be skipped. diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/40-gitlab.md b/docs/versioned_docs/version-2.6/30-administration/11-forges/40-gitlab.md similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/40-gitlab.md rename to docs/versioned_docs/version-2.6/30-administration/11-forges/40-gitlab.md diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/50-bitbucket.md b/docs/versioned_docs/version-2.6/30-administration/11-forges/50-bitbucket.md similarity index 97% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/50-bitbucket.md rename to docs/versioned_docs/version-2.6/30-administration/11-forges/50-bitbucket.md index b658238d6..d368e709c 100644 --- a/docs/versioned_docs/version-2.3/30-administration/11-forges/50-bitbucket.md +++ b/docs/versioned_docs/version-2.6/30-administration/11-forges/50-bitbucket.md @@ -14,7 +14,7 @@ WOODPECKER_BITBUCKET_SECRET=... ## Registration -You must register an OAuth application at Bitbucket in order to get a key and secret combination for woodpecker. Navigate to your workspace settings and choose `OAuth consumers` from the menu, and finally click `Add Consumer` (the url should be like: `https://bitbucket.org/[your-project-name]/workspace/settings/api`). +You must register an OAuth application at Bitbucket in order to get a key and secret combination for Woodpecker. Navigate to your workspace settings and choose `OAuth consumers` from the menu, and finally click `Add Consumer` (the url should be like: `https://bitbucket.org/[your-project-name]/workspace/settings/api`). Please set a name and set the `Callback URL` like this: diff --git a/docs/versioned_docs/version-2.6/30-administration/11-forges/60-bitbucket_datacenter.md b/docs/versioned_docs/version-2.6/30-administration/11-forges/60-bitbucket_datacenter.md new file mode 100644 index 000000000..9304d13a1 --- /dev/null +++ b/docs/versioned_docs/version-2.6/30-administration/11-forges/60-bitbucket_datacenter.md @@ -0,0 +1,98 @@ +--- +toc_max_heading_level: 2 +--- + +# Bitbucket Datacenter / Server + +:::warning +Woodpecker comes with experimental support for Bitbucket Datacenter / Server, formerly known as Atlassian Stash. +::: + +To enable Bitbucket Server you should configure the Woodpecker container using the following environment variables: + +```diff title="docker-compose.yaml" + version: '3' + + services: + woodpecker-server: + [...] + environment: + - [...] ++ - WOODPECKER_BITBUCKET_DC=true ++ - WOODPECKER_BITBUCKET_DC_GIT_USERNAME=foo ++ - WOODPECKER_BITBUCKET_DC_GIT_PASSWORD=bar ++ - WOODPECKER_BITBUCKET_DC_CLIENT_ID=xxx ++ - WOODPECKER_BITBUCKET_DC_CLIENT_SECRET=yyy ++ - WOODPECKER_BITBUCKET_DC_URL=http://stash.mycompany.com + + woodpecker-agent: + [...] +``` + +## Service Account + +Woodpecker uses `git+https` to clone repositories, however, Bitbucket Server does not currently support cloning repositories with an OAuth token. To work around this limitation, you must create a service account and provide the username and password to Woodpecker. This service account will be used to authenticate and clone private repositories. + +## Registration + +Woodpecker must be registered with Bitbucket Datacenter / Server. In the administration section of Bitbucket choose "Application Links" and then "Create link". Woodpecker should be listed as "External Application" and the direction should be set to "Incomming". Note the client id and client secret of the registration to be used in the configuration of Woodpecker. + +See also [Configure an incoming link](https://confluence.atlassian.com/bitbucketserver/configure-an-incoming-link-1108483657.html). + +## Configuration + +This is a full list of configuration options. Please note that many of these options use default configuration values that should work for the majority of installations. + +### `WOODPECKER_BITBUCKET_DC` + +> Default: `false` + +Enables the Bitbucket Server driver. + +### `WOODPECKER_BITBUCKET_DC_URL` + +> Default: empty + +Configures the Bitbucket Server address. + +### `WOODPECKER_BITBUCKET_DC_CLIENT_ID` + +> Default: empty + +Configures your Bitbucket Server OAUth 2.0 client id. + +### `WOODPECKER_BITBUCKET_DC_CLIENT_SECRET` + +> Default: empty + +Configures your Bitbucket Server OAUth 2.0 client secret. + +### `WOODPECKER_BITBUCKET_DC_GIT_USERNAME` + +> Default: empty + +This username is used to authenticate and clone all private repositories. + +### `WOODPECKER_BITBUCKET_DC_GIT_USERNAME_FILE` + +> Default: empty + +Read the value for `WOODPECKER_BITBUCKET_DC_GIT_USERNAME` from the specified filepath + +### `WOODPECKER_BITBUCKET_DC_GIT_PASSWORD` + +> Default: empty + +The password is used to authenticate and clone all private repositories. + +### `WOODPECKER_BITBUCKET_DC_GIT_PASSWORD_FILE` + +> Default: empty + +Read the value for `WOODPECKER_BITBUCKET_DC_GIT_PASSWORD` from the specified filepath + +### `WOODPECKER_BITBUCKET_DC_SKIP_VERIFY` + +> Default: `false` + +Configure if SSL verification should be skipped. diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/_category_.yaml b/docs/versioned_docs/version-2.6/30-administration/11-forges/_category_.yaml similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/_category_.yaml rename to docs/versioned_docs/version-2.6/30-administration/11-forges/_category_.yaml diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/bitbucket_oauth.png b/docs/versioned_docs/version-2.6/30-administration/11-forges/bitbucket_oauth.png similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/bitbucket_oauth.png rename to docs/versioned_docs/version-2.6/30-administration/11-forges/bitbucket_oauth.png diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/bitbucket_permissions.png b/docs/versioned_docs/version-2.6/30-administration/11-forges/bitbucket_permissions.png similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/bitbucket_permissions.png rename to docs/versioned_docs/version-2.6/30-administration/11-forges/bitbucket_permissions.png diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/gitea_oauth.gif b/docs/versioned_docs/version-2.6/30-administration/11-forges/gitea_oauth.gif similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/gitea_oauth.gif rename to docs/versioned_docs/version-2.6/30-administration/11-forges/gitea_oauth.gif diff --git a/docs/versioned_docs/version-2.3/30-administration/11-forges/github_oauth.png b/docs/versioned_docs/version-2.6/30-administration/11-forges/github_oauth.png similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/11-forges/github_oauth.png rename to docs/versioned_docs/version-2.6/30-administration/11-forges/github_oauth.png diff --git a/docs/versioned_docs/version-2.3/30-administration/15-agent-config.md b/docs/versioned_docs/version-2.6/30-administration/15-agent-config.md similarity index 98% rename from docs/versioned_docs/version-2.3/30-administration/15-agent-config.md rename to docs/versioned_docs/version-2.6/30-administration/15-agent-config.md index 4a3a5e5e0..15792b38b 100644 --- a/docs/versioned_docs/version-2.3/30-administration/15-agent-config.md +++ b/docs/versioned_docs/version-2.6/30-administration/15-agent-config.md @@ -168,12 +168,6 @@ Configures if the gRPC server certificate should be verified, only valid when `W Configures the backend engine to run pipelines on. Possible values are `auto-detect`, `docker`, `local` or `kubernetes`. -### `WOODPECKER_ADDONS` - -> Default: empty - -List of addon files. See [addons](./75-addons/00-overview.md). - ### `WOODPECKER_BACKEND_DOCKER_*` See [Docker backend configuration](./22-backends/10-docker.md#configuration) diff --git a/docs/versioned_docs/version-2.3/30-administration/22-backends/10-docker.md b/docs/versioned_docs/version-2.6/30-administration/22-backends/10-docker.md similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/22-backends/10-docker.md rename to docs/versioned_docs/version-2.6/30-administration/22-backends/10-docker.md diff --git a/docs/versioned_docs/version-2.3/30-administration/22-backends/20-local.md b/docs/versioned_docs/version-2.6/30-administration/22-backends/20-local.md similarity index 54% rename from docs/versioned_docs/version-2.3/30-administration/22-backends/20-local.md rename to docs/versioned_docs/version-2.6/30-administration/22-backends/20-local.md index 4dca5d34e..c0faf9c2f 100644 --- a/docs/versioned_docs/version-2.3/30-administration/22-backends/20-local.md +++ b/docs/versioned_docs/version-2.6/30-administration/22-backends/20-local.md @@ -5,33 +5,31 @@ toc_max_heading_level: 3 # Local backend :::danger -The local backend will execute the pipelines on the local system without any isolation of any kind. +The local backend executes pipelines on the local system without any isolation. ::: :::note -Currently we do not support services for this backend. +Currently we do not support [services](../../20-usage/60-services.md) for this backend. [Read more here](https://github.com/woodpecker-ci/woodpecker/issues/3095). ::: -Since the code runs directly in the same context as the agent (same user, same +Since the commands run directly in the same context as the agent (same user, same filesystem), a malicious pipeline could be used to access the agent configuration especially the `WOODPECKER_AGENT_SECRET` variable. It is recommended to use this backend only for private setup where the code and -pipeline can be trusted. You shouldn't use it for a public facing CI where -anyone can submit code or add new repositories. You shouldn't execute the agent -as a privileged user (root). +pipeline can be trusted. It should not be used in a public instance where +anyone can submit code or add new repositories. The agent should not run as a privileged user (root). -The local backend will use a random directory in $TMPDIR to store the cloned +The local backend will use a random directory in `$TMPDIR` to store the cloned code and execute commands. In order to use this backend, you need to download (or build) the -[binary](https://github.com/woodpecker-ci/woodpecker/releases/latest) of the -agent, configure it and run it on the host machine. +[agent](https://github.com/woodpecker-ci/woodpecker/releases/latest), configure it and run it on the host machine. ## Usage -To enable the local backend, add this to your configuration: +To enable the local backend, set the following: ```ini WOODPECKER_BACKEND=local @@ -39,7 +37,7 @@ WOODPECKER_BACKEND=local ### Shell -The `image` entry is used to specify the shell, such as Bash or Fish, that is +The `image` entrypoint is used to specify the shell, such as `bash` or `fish`, that is used to run the commands. ```yaml title=".woodpecker.yaml" @@ -51,15 +49,13 @@ steps: ### Plugins -Plugins are just executable binaries: - ```yaml steps: - name: build image: /usr/bin/tree ``` -If no commands are provided, we treat them as plugins in the usual manner. +If no commands are provided, plugins are treated in the usual manner. In the context of the local backend, plugins are simply executable binaries, which can be located using their name if they are listed in `$PATH`, or through an absolute path. ### Options diff --git a/docs/versioned_docs/version-2.6/30-administration/22-backends/40-kubernetes.md b/docs/versioned_docs/version-2.6/30-administration/22-backends/40-kubernetes.md new file mode 100644 index 000000000..48f4a622d --- /dev/null +++ b/docs/versioned_docs/version-2.6/30-administration/22-backends/40-kubernetes.md @@ -0,0 +1,305 @@ +--- +toc_max_heading_level: 2 +--- + +# Kubernetes backend + +The Kubernetes backend executes steps inside standalone Pods. A temporary PVC is created for the lifetime of the pipeline to transfer files between steps. + +## Images from private registries + +In order to pull private container images defined in your pipeline YAML you must provide [registry credentials in Kubernetes Secret](https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/). +As the Secret is Agent-wide, it has to be placed in namespace defined by `WOODPECKER_BACKEND_K8S_NAMESPACE`. +Besides, you need to provide the Secret name to Agent via `WOODPECKER_BACKEND_K8S_PULL_SECRET_NAMES`. + +## Job specific configuration + +### Resources + +The Kubernetes backend also allows for specifying requests and limits on a per-step basic, most commonly for CPU and memory. +We recommend to add a `resources` definition to all steps to ensure efficient scheduling. + +Here is an example definition with an arbitrary `resources` definition below the `backend_options` section: + +```yaml +steps: + - name: 'My kubernetes step' + image: alpine + commands: + - echo "Hello world" + backend_options: + kubernetes: + resources: + requests: + memory: 200Mi + cpu: 100m + limits: + memory: 400Mi + cpu: 1000m +``` + +You can use [Limit Ranges](https://kubernetes.io/docs/concepts/policy/limit-range/) if you want to set the limits by per-namespace basis. + +### Runtime class + +`runtimeClassName` specifies the name of the RuntimeClass which will be used to run this Pod. If no `runtimeClassName` is specified, the default RuntimeHandler will be used. +See the [Kubernetes documentation](https://kubernetes.io/docs/concepts/containers/runtime-class/) for more information on specifying runtime classes. + +### Service account + +`serviceAccountName` specifies the name of the ServiceAccount which the Pod will mount. This service account must be created externally. +See the [Kubernetes documentation](https://kubernetes.io/docs/concepts/security/service-accounts/) for more information on using service accounts. + +### Node selector + +`nodeSelector` specifies the labels which are used to select the node on which the job will be executed. + +Labels defined here will be appended to a list which already contains `"kubernetes.io/arch"`. +By default `"kubernetes.io/arch"` is inferred from the agents' platform. One can override it by setting that label in the `nodeSelector` section of the `backend_options`. +Without a manual overwrite, builds will be randomly assigned to the runners and inherit their respective architectures. + +To overwrite this, one needs to set the label in the `nodeSelector` section of the `backend_options`. +A practical example for this is when running a matrix-build and delegating specific elements of the matrix to run on a specific architecture. +In this case, one must define an arbitrary key in the matrix section of the respective matrix element: + +```yaml +matrix: + include: + - NAME: runner1 + ARCH: arm64 +``` + +And then overwrite the `nodeSelector` in the `backend_options` section of the step(s) using the name of the respective env var: + +```yaml +[...] + backend_options: + kubernetes: + nodeSelector: + kubernetes.io/arch: "${ARCH}" +``` + +You can use [WOODPECKER_BACKEND_K8S_POD_NODE_SELECTOR](#woodpecker_backend_k8s_pod_node_selector) if you want to set the node selector per Agent +or [PodNodeSelector](https://kubernetes.io/docs/reference/access-authn-authz/admission-controllers/#podnodeselector) admission controller if you want to set the node selector by per-namespace basis. + +### Tolerations + +When you use `nodeSelector` and the node pool is configured with Taints, you need to specify the Tolerations. Tolerations allow the scheduler to schedule Pods with matching taints. +See the [Kubernetes documentation](https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/) for more information on using tolerations. + +Example pipeline configuration: + +```yaml +steps: + - name: build + image: golang + commands: + - go get + - go build + - go test + backend_options: + kubernetes: + serviceAccountName: 'my-service-account' + resources: + requests: + memory: 128Mi + cpu: 1000m + limits: + memory: 256Mi + nodeSelector: + beta.kubernetes.io/instance-type: p3.8xlarge + tolerations: + - key: 'key1' + operator: 'Equal' + value: 'value1' + effect: 'NoSchedule' + tolerationSeconds: 3600 +``` + +### Volumes + +To mount volumes a PersistentVolume (PV) and PersistentVolumeClaim (PVC) are needed on the cluster which can be referenced in steps via the `volumes` option. +Assuming a PVC named `woodpecker-cache` exists, it can be referenced as follows in a step: + +```yaml +steps: + - name: "Restore Cache" + image: meltwater/drone-cache + volumes: + - woodpecker-cache:/woodpecker/src/cache + settings: + mount: + - "woodpecker-cache" + [...] +``` + +### Security context + +Use the following configuration to set the [Security Context](https://kubernetes.io/docs/tasks/configure-pod-container/security-context/) for the Pod/container running a given pipeline step: + +```yaml +steps: + - name: test + image: alpine + commands: + - echo Hello world + backend_options: + kubernetes: + securityContext: + runAsUser: 999 + runAsGroup: 999 + privileged: true + [...] +``` + +Note that the `backend_options.kubernetes.securityContext` object allows you to set both Pod and container level security context options in one object. +By default, the properties will be set at the Pod level. Properties that are only supported on the container level will be set there instead. So, the +configuration shown above will result in something like the following Pod spec: + +```yaml +kind: Pod +spec: + securityContext: + runAsUser: 999 + runAsGroup: 999 + containers: + - name: wp-01hcd83q7be5ymh89k5accn3k6-0-step-0 + image: alpine + securityContext: + privileged: true + [...] +``` + +You can also restrict a container's syscalls with [seccomp](https://kubernetes.io/docs/tutorials/security/seccomp/) profile + +```yaml +backend_options: + kubernetes: + securityContext: + seccompProfile: + type: Localhost + localhostProfile: profiles/audit.json +``` + +or restrict a container's access to resources by specifying [AppArmor](https://kubernetes.io/docs/tutorials/security/apparmor/) profile + +```yaml +backend_options: + kubernetes: + securityContext: + apparmorProfile: + type: Localhost + localhostProfile: k8s-apparmor-example-deny-write +``` + +:::note +AppArmor syntax follows [KEP-24](https://github.com/kubernetes/enhancements/blob/fddcbb9cbf3df39ded03bad71228265ac6e5215f/keps/sig-node/24-apparmor/README.md). +::: + +### Annotations and labels + +You can specify arbitrary [annotations](https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/) and [labels](https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/) to be set on the Pod definition for a given workflow step using the following configuration: + +```yaml +backend_options: + kubernetes: + annotations: + workflow-group: alpha + io.kubernetes.cri-o.Devices: /dev/fuse + labels: + environment: ci + app.kubernetes.io/name: builder +``` + +In order to enable this configuration you need to set the appropriate environment variables to `true` on the woodpecker agent: +[WOODPECKER_BACKEND_K8S_POD_ANNOTATIONS_ALLOW_FROM_STEP](#woodpecker_backend_k8s_pod_annotations_allow_from_step) and/or [WOODPECKER_BACKEND_K8S_POD_LABELS_ALLOW_FROM_STEP](#woodpecker_backend_k8s_pod_labels_allow_from_step). + +## Tips and tricks + +### CRI-O + +CRI-O users currently need to configure the workspace for all workflows in order for them to run correctly. Add the following at the beginning of your configuration: + +```yaml +workspace: + base: '/woodpecker' + path: '/' +``` + +See [this issue](https://github.com/woodpecker-ci/woodpecker/issues/2510) for more details. + +### `KUBERNETES_SERVICE_HOST` environment variable + +Like the below env vars used for configuration, this can be set in the environment fonfiguration of the agent. It configures the address of the Kubernetes API server to connect to. + +If running the agent within Kubernetes, this will already be set and you don't have to add it manually. + +## Configuration + +These env vars can be set in the `env:` sections of the agent. + +### `WOODPECKER_BACKEND_K8S_NAMESPACE` + +> Default: `woodpecker` + +The namespace to create worker Pods in. + +### `WOODPECKER_BACKEND_K8S_VOLUME_SIZE` + +> Default: `10G` + +The volume size of the pipeline volume. + +### `WOODPECKER_BACKEND_K8S_STORAGE_CLASS` + +> Default: empty + +The storage class to use for the pipeline volume. + +### `WOODPECKER_BACKEND_K8S_STORAGE_RWX` + +> Default: `true` + +Determines if `RWX` should be used for the pipeline volume's [access mode](https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes). If false, `RWO` is used instead. + +### `WOODPECKER_BACKEND_K8S_POD_LABELS` + +> Default: empty + +Additional labels to apply to worker Pods. Must be a YAML object, e.g. `{"example.com/test-label":"test-value"}`. + +### `WOODPECKER_BACKEND_K8S_POD_LABELS_ALLOW_FROM_STEP` + +> Default: `false` + +Determines if additional Pod labels can be defined from a step's backend options. + +### `WOODPECKER_BACKEND_K8S_POD_ANNOTATIONS` + +> Default: empty + +Additional annotations to apply to worker Pods. Must be a YAML object, e.g. `{"example.com/test-annotation":"test-value"}`. + +### `WOODPECKER_BACKEND_K8S_POD_ANNOTATIONS_ALLOW_FROM_STEP` + +> Default: `false` + +Determines if Pod annotations can be defined from a step's backend options. + +### `WOODPECKER_BACKEND_K8S_POD_NODE_SELECTOR` + +> Default: empty + +Additional node selector to apply to worker pods. Must be a YAML object, e.g. `{"topology.kubernetes.io/region":"eu-central-1"}`. + +### `WOODPECKER_BACKEND_K8S_SECCTX_NONROOT` + +> Default: `false` + +Determines if containers must be required to run as non-root users. + +### `WOODPECKER_BACKEND_K8S_PULL_SECRET_NAMES` + +> Default: empty + +Secret names to pull images from private repositories. See, how to [Pull an Image from a Private Registry](https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/). diff --git a/docs/versioned_docs/version-2.6/30-administration/22-backends/50-custom-backends.md b/docs/versioned_docs/version-2.6/30-administration/22-backends/50-custom-backends.md new file mode 100644 index 000000000..3c771c4ef --- /dev/null +++ b/docs/versioned_docs/version-2.6/30-administration/22-backends/50-custom-backends.md @@ -0,0 +1,23 @@ +# Custom backends + +If none of our backends fits your usecases, you can write your own. + +Therefore, implement the interface `"go.woodpecker-ci.org/woodpecker/woodpecker/v2/pipeline/backend/types".Backend` and +build a custom agent using your backend with this `main.go`: + +```go +package main + +import ( + "go.woodpecker-ci.org/woodpecker/v2/cmd/agent/core" + backendTypes "go.woodpecker-ci.org/woodpecker/v2/pipeline/backend/types" +) + +func main() { + core.RunAgent([]backendTypes.Backend{ + yourBackend, + }) +} +``` + +It is also possible to use multiple backends, you can select with [`WOODPECKER_BACKEND`](../15-agent-config.md#woodpecker_backend) between them. diff --git a/docs/versioned_docs/version-2.3/30-administration/22-backends/_category_.yaml b/docs/versioned_docs/version-2.6/30-administration/22-backends/_category_.yaml similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/22-backends/_category_.yaml rename to docs/versioned_docs/version-2.6/30-administration/22-backends/_category_.yaml diff --git a/docs/versioned_docs/version-2.3/30-administration/30-database.md b/docs/versioned_docs/version-2.6/30-administration/30-database.md similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/30-database.md rename to docs/versioned_docs/version-2.6/30-administration/30-database.md diff --git a/docs/versioned_docs/version-2.3/30-administration/60-ssl.md b/docs/versioned_docs/version-2.6/30-administration/60-ssl.md similarity index 98% rename from docs/versioned_docs/version-2.3/30-administration/60-ssl.md rename to docs/versioned_docs/version-2.6/30-administration/60-ssl.md index 74de21023..755ba205d 100644 --- a/docs/versioned_docs/version-2.3/30-administration/60-ssl.md +++ b/docs/versioned_docs/version-2.6/30-administration/60-ssl.md @@ -78,7 +78,7 @@ Update your configuration to mount your certificate and key: Update your configuration to provide the paths of your certificate and key: -```yaml title="docker-compose.yaml" +```diff title="docker-compose.yaml" version: '3' services: diff --git a/docs/versioned_docs/version-2.3/30-administration/70-proxy.md b/docs/versioned_docs/version-2.6/30-administration/70-proxy.md similarity index 99% rename from docs/versioned_docs/version-2.3/30-administration/70-proxy.md rename to docs/versioned_docs/version-2.6/30-administration/70-proxy.md index 4aae2fbe1..1e253e457 100644 --- a/docs/versioned_docs/version-2.3/30-administration/70-proxy.md +++ b/docs/versioned_docs/version-2.6/30-administration/70-proxy.md @@ -31,7 +31,7 @@ You must configure Apache to set `X-Forwarded-Proto` when using https. ## Nginx -This guide provides a basic overview for installing Woodpecker server behind the Nginx web-server. For more advanced configuration options please consult the official Nginx [documentation](https://www.nginx.com/resources/admin-guide/). +This guide provides a basic overview for installing Woodpecker server behind the Nginx web-server. For more advanced configuration options please consult the official Nginx [documentation](https://docs.nginx.com/nginx/admin-guide). Example configuration: diff --git a/docs/versioned_docs/version-2.3/30-administration/80-autoscaler.md b/docs/versioned_docs/version-2.6/30-administration/80-autoscaler.md similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/80-autoscaler.md rename to docs/versioned_docs/version-2.6/30-administration/80-autoscaler.md diff --git a/docs/versioned_docs/version-2.3/30-administration/90-prometheus.md b/docs/versioned_docs/version-2.6/30-administration/90-prometheus.md similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/90-prometheus.md rename to docs/versioned_docs/version-2.6/30-administration/90-prometheus.md diff --git a/docs/versioned_docs/version-2.3/30-administration/_category_.yaml b/docs/versioned_docs/version-2.6/30-administration/_category_.yaml similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/_category_.yaml rename to docs/versioned_docs/version-2.6/30-administration/_category_.yaml diff --git a/docs/versioned_docs/version-2.3/30-administration/new-agent-connected.png b/docs/versioned_docs/version-2.6/30-administration/new-agent-connected.png similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/new-agent-connected.png rename to docs/versioned_docs/version-2.6/30-administration/new-agent-connected.png diff --git a/docs/versioned_docs/version-2.3/30-administration/new-agent-created.png b/docs/versioned_docs/version-2.6/30-administration/new-agent-created.png similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/new-agent-created.png rename to docs/versioned_docs/version-2.6/30-administration/new-agent-created.png diff --git a/docs/versioned_docs/version-2.3/30-administration/new-agent-registration.png b/docs/versioned_docs/version-2.6/30-administration/new-agent-registration.png similarity index 100% rename from docs/versioned_docs/version-2.3/30-administration/new-agent-registration.png rename to docs/versioned_docs/version-2.6/30-administration/new-agent-registration.png diff --git a/docs/versioned_docs/version-2.6/40-cli.md b/docs/versioned_docs/version-2.6/40-cli.md new file mode 100644 index 000000000..19a051967 --- /dev/null +++ b/docs/versioned_docs/version-2.6/40-cli.md @@ -0,0 +1,594 @@ +# CLI + +# NAME + +woodpecker-cli - A new cli application + +# SYNOPSIS + +woodpecker-cli + +``` +[--config|-c]=[value] +[--disable-update-check] +[--log-file]=[value] +[--log-level]=[value] +[--nocolor] +[--pretty] +[--server|-s]=[value] +[--token|-t]=[value] +``` + +# DESCRIPTION + +Woodpecker command line utility + +**Usage**: + +``` +woodpecker-cli [GLOBAL OPTIONS] command [COMMAND OPTIONS] [ARGUMENTS...] +``` + +# GLOBAL OPTIONS + +**--config, -c**="": path to config file + +**--disable-update-check**: disable update check + +**--log-file**="": Output destination for logs. 'stdout' and 'stderr' can be used as special keywords. (default: "stderr") + +**--log-level**="": set logging level (default: "info") + +**--nocolor**: disable colored debug output, only has effect if pretty output is set too + +**--pretty**: enable pretty-printed debug output + +**--server, -s**="": server address + +**--token, -t**="": server auth token + +# COMMANDS + +## pipeline + +manage pipelines + +### ls + +show pipeline history + +**--branch**="": branch filter + +**--event**="": event filter + +**--format**="": format output (default: "\x1b[33mPipeline #{{ .Number }} \x1b[0m\nStatus: {{ .Status }}\nEvent: {{ .Event }}\nCommit: {{ .Commit }}\nBranch: {{ .Branch }}\nRef: {{ .Ref }}\nAuthor: {{ .Author }} {{ if .Email }}<{{.Email}}>{{ end }}\nMessage: {{ .Message }}\n") + +**--limit**="": limit the list size (default: 25) + +**--status**="": status filter + +### last + +show latest pipeline details + +**--branch**="": branch name (default: "main") + +**--format**="": format output (default: "Number: {{ .Number }}\nStatus: {{ .Status }}\nEvent: {{ .Event }}\nCommit: {{ .Commit }}\nBranch: {{ .Branch }}\nRef: {{ .Ref }}\nMessage: {{ .Message }}\nAuthor: {{ .Author }}\n") + +### logs + +show pipeline logs + +### info + +show pipeline details + +**--format**="": format output (default: "Number: {{ .Number }}\nStatus: {{ .Status }}\nEvent: {{ .Event }}\nCommit: {{ .Commit }}\nBranch: {{ .Branch }}\nRef: {{ .Ref }}\nMessage: {{ .Message }}\nAuthor: {{ .Author }}\n") + +### stop + +stop a pipeline + +### start + +start a pipeline + +**--param, -p**="": custom parameters to be injected into the step environment. Format: KEY=value + +### approve + +approve a pipeline + +### decline + +decline a pipeline + +### queue + +show pipeline queue + +**--format**="": format output (default: "\x1b[33m{{ .FullName }} #{{ .Number }} \x1b[0m\nStatus: {{ .Status }}\nEvent: {{ .Event }}\nCommit: {{ .Commit }}\nBranch: {{ .Branch }}\nRef: {{ .Ref }}\nAuthor: {{ .Author }} {{ if .Email }}<{{.Email}}>{{ end }}\nMessage: {{ .Message }}\n") + +### ps + +show pipeline steps + +**--format**="": format output (default: "\x1b[33mStep #{{ .PID }} \x1b[0m\nStep: {{ .Name }}\nState: {{ .State }}\n") + +### create + +create new pipeline + +**--branch**="": branch to create pipeline from + +**--format**="": format output (default: "\x1b[33mPipeline #{{ .Number }} \x1b[0m\nStatus: {{ .Status }}\nEvent: {{ .Event }}\nCommit: {{ .Commit }}\nBranch: {{ .Branch }}\nRef: {{ .Ref }}\nAuthor: {{ .Author }} {{ if .Email }}<{{.Email}}>{{ end }}\nMessage: {{ .Message }}\n") + +**--var**="": key=value + +## log + +manage logs + +### purge + +purge a log + +## deploy + +deploy code + +**--branch**="": branch filter (default: "main") + +**--event**="": event filter (default: "push") + +**--format**="": format output (default: "Number: {{ .Number }}\nStatus: {{ .Status }}\nCommit: {{ .Commit }}\nBranch: {{ .Branch }}\nRef: {{ .Ref }}\nMessage: {{ .Message }}\nAuthor: {{ .Author }}\nTarget: {{ .Deploy }}\n") + +**--param, -p**="": custom parameters to be injected into the step environment. Format: KEY=value + +**--status**="": status filter (default: "success") + +## exec + +execute a local pipeline + +**--backend-docker-api-version**="": the version of the API to reach, leave empty for latest. + +**--backend-docker-cert**="": path to load the TLS certificates for connecting to docker server + +**--backend-docker-host**="": path to docker socket or url to the docker server + +**--backend-docker-ipv6**: backend docker enable IPV6 + +**--backend-docker-network**="": backend docker network + +**--backend-docker-tls-verify**: enable or disable TLS verification for connecting to docker server + +**--backend-docker-volumes**="": backend docker volumes (comma separated) + +**--backend-engine**="": backend engine to run pipelines on (default: "auto-detect") + +**--backend-http-proxy**="": if set, pass the environment variable down as "HTTP_PROXY" to steps + +**--backend-https-proxy**="": if set, pass the environment variable down as "HTTPS_PROXY" to steps + +**--backend-k8s-namespace**="": backend k8s namespace (default: "woodpecker") + +**--backend-k8s-pod-annotations**="": backend k8s additional worker pod annotations + +**--backend-k8s-pod-image-pull-secret-names**="": backend k8s pull secret names for private registries (default: "regcred") + +**--backend-k8s-pod-labels**="": backend k8s additional worker pod labels + +**--backend-k8s-secctx-nonroot**: `run as non root` Kubernetes security context option + +**--backend-k8s-storage-class**="": backend k8s storage class + +**--backend-k8s-storage-rwx**: backend k8s storage access mode, should ReadWriteMany (RWX) instead of ReadWriteOnce (RWO) be used? (default: true) + +**--backend-k8s-volume-size**="": backend k8s volume size (default 10G) (default: "10G") + +**--backend-local-temp-dir**="": set a different temp dir to clone workflows into (default: "/tmp") + +**--backend-no-proxy**="": if set, pass the environment variable down as "NO_PROXY" to steps + +**--commit-author-avatar**="": + +**--commit-author-email**="": + +**--commit-author-name**="": + +**--commit-branch**="": + +**--commit-message**="": + +**--commit-ref**="": + +**--commit-refspec**="": + +**--commit-sha**="": + +**--env**="": + +**--forge-type**="": + +**--forge-url**="": + +**--local**: run from local directory + +**--netrc-machine**="": + +**--netrc-password**="": + +**--netrc-username**="": + +**--network**="": external networks + +**--pipeline-created**="": (default: 0) + +**--pipeline-event**="": (default: "manual") + +**--pipeline-finished**="": (default: 0) + +**--pipeline-number**="": (default: 0) + +**--pipeline-parent**="": (default: 0) + +**--pipeline-started**="": (default: 0) + +**--pipeline-status**="": + +**--pipeline-target**="": + +**--pipeline-url**="": + +**--prev-commit-author-avatar**="": + +**--prev-commit-author-email**="": + +**--prev-commit-author-name**="": + +**--prev-commit-branch**="": + +**--prev-commit-message**="": + +**--prev-commit-ref**="": + +**--prev-commit-refspec**="": + +**--prev-commit-sha**="": + +**--prev-pipeline-created**="": (default: 0) + +**--prev-pipeline-event**="": + +**--prev-pipeline-finished**="": (default: 0) + +**--prev-pipeline-number**="": (default: 0) + +**--prev-pipeline-started**="": (default: 0) + +**--prev-pipeline-status**="": + +**--prev-pipeline-url**="": + +**--privileged**="": privileged plugins (default: "plugins/docker", "plugins/gcr", "plugins/ecr", "woodpeckerci/plugin-docker-buildx", "codeberg.org/woodpecker-plugins/docker-buildx") + +**--repo**="": full repo name + +**--repo-clone-ssh-url**="": + +**--repo-clone-url**="": + +**--repo-path**="": path to local repository + +**--repo-private**="": + +**--repo-remote-id**="": + +**--repo-trusted**: + +**--repo-url**="": + +**--step-name**="": (default: 0) + +**--system-name**="": (default: "woodpecker") + +**--system-platform**="": + +**--system-url**="": (default: "https://github.com/woodpecker-ci/woodpecker") + +**--timeout**="": pipeline timeout (default: 1h0m0s) + +**--volumes**="": pipeline volumes + +**--workflow-name**="": (default: 0) + +**--workflow-number**="": (default: 0) + +**--workspace-base**="": (default: "/woodpecker") + +**--workspace-path**="": (default: "src") + +## info + +show information about the current user + +## registry + +manage registries + +### add + +adds a registry + +**--hostname**="": registry hostname (default: "docker.io") + +**--password**="": registry password + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +**--username**="": registry username + +### rm + +remove a registry + +**--hostname**="": registry hostname (default: "docker.io") + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +### update + +update a registry + +**--hostname**="": registry hostname (default: "docker.io") + +**--password**="": registry password + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +**--username**="": registry username + +### info + +display registry info + +**--hostname**="": registry hostname (default: "docker.io") + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +### ls + +list registries + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +## secret + +manage secrets + +### add + +adds a secret + +**--event**="": secret limited to these events + +**--global**: global secret + +**--image**="": secret limited to these images + +**--name**="": secret name + +**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +**--value**="": secret value + +### rm + +remove a secret + +**--global**: global secret + +**--name**="": secret name + +**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +### update + +update a secret + +**--event**="": secret limited to these events + +**--global**: global secret + +**--image**="": secret limited to these images + +**--name**="": secret name + +**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +**--value**="": secret value + +### info + +display secret info + +**--global**: global secret + +**--name**="": secret name + +**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +### ls + +list secrets + +**--global**: global secret + +**--organization, --org**="": organization id or full-name (e.g. 123 or octocat) + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +## repo + +manage repositories + +### ls + +list all repos + +**--format**="": format output (default: "\x1b[33m{{ .FullName }}\x1b[0m (id: {{ .ID }}, forgeRemoteID: {{ .ForgeRemoteID }})") + +**--org**="": filter by organization + +### info + +show repository details + +**--format**="": format output (default: "Owner: {{ .Owner }}\nRepo: {{ .Name }}\nURL: {{ .ForgeURL }}\nConfig path: {{ .Config }}\nVisibility: {{ .Visibility }}\nPrivate: {{ .IsSCMPrivate }}\nTrusted: {{ .IsTrusted }}\nGated: {{ .IsGated }}\nClone url: {{ .Clone }}\nAllow pull-requests: {{ .AllowPullRequests }}\n") + +### add + +add a repository + +### update + +update a repository + +**--config**="": repository configuration path (e.g. .woodpecker.yml) + +**--gated**: repository is gated + +**--pipeline-counter**="": repository starting pipeline number (default: 0) + +**--timeout**="": repository timeout (default: 0s) + +**--trusted**: repository is trusted + +**--unsafe**: validate updating the pipeline-counter is unsafe + +**--visibility**="": repository visibility + +### rm + +remove a repository + +### repair + +repair repository webhooks + +### chown + +assume ownership of a repository + +### sync + +synchronize the repository list + +**--format**="": format output (default: "\x1b[33m{{ .FullName }}\x1b[0m (id: {{ .ID }}, forgeRemoteID: {{ .ForgeRemoteID }})") + +## user + +manage users + +### ls + +list all users + +**--format**="": format output (default: "{{ .Login }}") + +### info + +show user details + +**--format**="": format output (default: "User: {{ .Login }}\nEmail: {{ .Email }}") + +### add + +adds a user + +### rm + +remove a user + +## lint + +lint a pipeline configuration file + +## log-level + +get the logging level of the server, or set it with [level] + +## cron + +manage cron jobs + +### add + +add a cron job + +**--branch**="": cron branch + +**--name**="": cron name + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +**--schedule**="": cron schedule + +### rm + +remove a cron job + +**--id**="": cron id + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +### update + +update a cron job + +**--branch**="": cron branch + +**--id**="": cron id + +**--name**="": cron name + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +**--schedule**="": cron schedule + +### info + +display info about a cron job + +**--id**="": cron id + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +### ls + +list cron jobs + +**--repository, --repo**="": repository id or full-name (e.g. 134 or octocat/hello-world) + +## setup + +setup the woodpecker-cli for the first time + +**--server-url**="": The URL of the woodpecker server + +**--token**="": The token to authenticate with the woodpecker server + +## update + +update the woodpecker-cli to the latest version + +**--force**: force update even if the latest version is already installed diff --git a/docs/versioned_docs/version-2.6/50-about.md b/docs/versioned_docs/version-2.6/50-about.md new file mode 100644 index 000000000..bec3304a1 --- /dev/null +++ b/docs/versioned_docs/version-2.6/50-about.md @@ -0,0 +1,18 @@ +# About + +Woodpecker has been originally forked from Drone 0.8 as the Drone CI license was changed after the 0.8 release from Apache 2.0 to a proprietary license. Woodpecker is based on this latest freely available version. + +## History + +Woodpecker was originally forked by [@laszlocph](https://github.com/laszlocph) in 2019. + +A few important time points: + +- [`2fbaa56`](https://github.com/woodpecker-ci/woodpecker/commit/2fbaa56eee0f4be7a3ca4be03dbd00c1bf5d1274) is the first commit of the fork, made on Apr 3, 2019. +- The first release [v0.8.91](https://github.com/woodpecker-ci/woodpecker/releases/tag/v0.8.91) was published on Apr 6, 2019. +- On Aug 27, 2019, the project was renamed to "Woodpecker" ([`630c383`](https://github.com/woodpecker-ci/woodpecker/commit/630c383181b10c4ec375e500c812c4b76b3c52b8)). +- The first release under the name "Woodpecker" was published on Sep 9, 2019 ([v0.8.104](https://github.com/woodpecker-ci/woodpecker/releases/tag/v0.8.104)). + +## Differences to Drone + +Woodpecker is a community-focused software that still stay free and open source forever, while Drone is managed by [Harness](https://harness.io/) and published under [Polyform Small Business](https://polyformproject.org/licenses/small-business/1.0.0/) license. diff --git a/docs/versioned_docs/version-2.3/91-migrations.md b/docs/versioned_docs/version-2.6/91-migrations.md similarity index 92% rename from docs/versioned_docs/version-2.3/91-migrations.md rename to docs/versioned_docs/version-2.6/91-migrations.md index 328d87a62..841dee3d0 100644 --- a/docs/versioned_docs/version-2.3/91-migrations.md +++ b/docs/versioned_docs/version-2.6/91-migrations.md @@ -2,11 +2,25 @@ Some versions need some changes to the server configuration or the pipeline configuration files. + + ## `next` - Deprecated `steps.[name].group` in favor of `steps.[name].depends_on` (see [workflow syntax](./20-usage/20-workflow-syntax.md#depends_on) to learn how to set dependencies) - Removed `WOODPECKER_ROOT_PATH` and `WOODPECKER_ROOT_URL` config variables. Use `WOODPECKER_HOST` with a path instead - Pipelines without a config file will now be skipped instead of failing +- Deprecated `includes` and `excludes` support from **event** filter +- Deprecated uppercasing all secret env vars, instead, the value of the `secrets` property is used. [Read more](./20-usage/40-secrets.md#use-secrets-in-commands) +- Deprecated alternative names for secrets, use `environment` with `from_secret` +- Deprecated slice definition for env vars +- Deprecated `environment` filter, use `when.evaluate` +- Use `WOODPECKER_EXPERT_FORGE_OAUTH_HOST` instead of `WOODPECKER_DEV_GITEA_OAUTH_URL` or `WOODPECKER_DEV_OAUTH_HOST` +- Deprecated `WOODPECKER_WEBHOOK_HOST` in favor of `WOODPECKER_EXPERT_WEBHOOK_HOST` ## 2.0.0 @@ -62,7 +76,7 @@ Some versions need some changes to the server configuration or the pipeline conf Only projects created after updating will have an empty value by default. Existing projects will stick to the current pipeline path which is `.drone.yml` in most cases. - Read more about it at the [Project Settings](./20-usage/71-project-settings.md#pipeline-path) + Read more about it at the [Project Settings](./20-usage/75-project-settings.md#pipeline-path) - From version `0.15.0` ongoing there will be three types of docker images: `latest`, `next` and `x.x.x` with an alpine variant for each type like `latest-alpine`. If you used `latest` before to try pre-release features you should switch to `next` after this release. diff --git a/docs/versioned_docs/version-2.3/92-awesome.md b/docs/versioned_docs/version-2.6/92-awesome.md similarity index 80% rename from docs/versioned_docs/version-2.3/92-awesome.md rename to docs/versioned_docs/version-2.6/92-awesome.md index 18f36ce71..920341d33 100644 --- a/docs/versioned_docs/version-2.3/92-awesome.md +++ b/docs/versioned_docs/version-2.6/92-awesome.md @@ -1,6 +1,6 @@ # Awesome Woodpecker -A curated list of awesome things related to Woodpecker-CI. +A curated list of awesome things related to Woodpecker CI. If you have some missing resources, please feel free to [open a pull-request](https://github.com/woodpecker-ci/woodpecker/edit/main/docs/docs/92-awesome.md) and add them. @@ -14,7 +14,7 @@ If you have some missing resources, please feel free to [open a pull-request](ht ## Projects using Woodpecker -- [Woodpecker-CI](https://github.com/woodpecker-ci/woodpecker/tree/main/.woodpecker) itself +- [Woodpecker CI](https://github.com/woodpecker-ci/woodpecker/tree/main/.woodpecker) itself - [All official plugins](https://github.com/woodpecker-ci?q=plugin&type=all) - [dessalines/thumb-key](https://github.com/dessalines/thumb-key/blob/main/.woodpecker.yml) - Android Jetpack compose linting and building - [Vieter](https://git.rustybever.be/vieter-v/vieter) - Archlinux/Pacman repository server & automated package build system @@ -24,12 +24,12 @@ If you have some missing resources, please feel free to [open a pull-request](ht ## Tools - [Convert Drone CI pipelines to Woodpecker CI](https://codeberg.org/lafriks/woodpecker-pipeline-transform) -- [Ansible NAS](https://github.com/davestephens/ansible-nas/) - a homelab Ansible playbook that can set up Woodpecker-CI and Gitea +- [Ansible NAS](https://github.com/davestephens/ansible-nas/) - a homelab Ansible playbook that can set up Woodpecker CI and Gitea - [picus](https://github.com/windsource/picus) - Picus connects to a Woodpecker CI server and creates an agent in the cloud when there are pending workflows. - [Hetzner cloud](https://www.hetzner.com/cloud) based [Woodpecker compatible autoscaler](https://git.ljoonal.xyz/ljoonal/hetzner-ci-autoscaler) - Creates and destroys VPS instances based on the count of pending & running jobs. -- [woodpecker-lint](https://git.schmidl.dev/schtobia/woodpecker-lint) - A repository for linting a woodpecker config file via pre-commit hook -- [Grafana Dashboard](https://github.com/Janik-Haag/woodpecker-grafana-dashboard) - A dashboard visualizing information exposed by the woodpecker prometheus endpoint. -- [woodpecker-autoscaler](https://github.com/Lerentis/woodpecker-autoscaler) - Yet another woodpecker autoscaler currently targeting [Hetzner cloud](https://www.hetzner.com/cloud) that works in parallel to other autoscaler implementations. +- [woodpecker-lint](https://git.schmidl.dev/schtobia/woodpecker-lint) - A repository for linting a Woodpecker config file via pre-commit hook +- [Grafana Dashboard](https://github.com/Janik-Haag/woodpecker-grafana-dashboard) - A dashboard visualizing information exposed by the Woodpecker prometheus endpoint. +- [woodpecker-autoscaler](https://github.com/Lerentis/woodpecker-autoscaler) - Yet another Woodpecker autoscaler currently targeting [Hetzner cloud](https://www.hetzner.com/cloud) that works in parallel to other autoscaler implementations. ## Configuration Services @@ -50,6 +50,11 @@ If you have some missing resources, please feel free to [open a pull-request](ht - [Locally Cached Nix CI with Woodpecker](https://blog.kotatsu.dev/posts/2023-04-21-woodpecker-nix-caching/) - [How to run Cypress auto-tests on Woodpecker CI and report results to Slack](https://devforth.io/blog/how-to-run-cypress-auto-tests-on-woodpecker-ci-and-report-results-to-slack/) - [Quest For CICD - WoodpeckerCI](https://omaramin.me/posts/woodpecker/) +- [Getting started with Woodpecker CI](https://systeemkabouter.eu/getting-started-with-woodpecker-ci.html) +- [Installing gitea and woodpecker using binary packages](https://neelex.com/2023/03/26/Installing-gitea-using-binary-packages/) +- [Deploying mdbook to codeberg pages using woodpecker CI](https://www.markpitblado.me/blog/deploying-mdbook-to-codeberg-pages-using-woodpecker-ci/) +- [Deploy a Fly app with Woodpecker CI](https://joeroe.io/2024/01/09/deploy-fly-woodpecker-ci.html) +- [Ansible - using Woodpecker as an alternative to Semaphore](https://pat-s.me/ansible-using-woodpecker-as-an-alternative-to-semaphore/) ## Videos diff --git a/docs/versioned_docs/version-2.3/92-development/01-getting-started.md b/docs/versioned_docs/version-2.6/92-development/01-getting-started.md similarity index 92% rename from docs/versioned_docs/version-2.3/92-development/01-getting-started.md rename to docs/versioned_docs/version-2.6/92-development/01-getting-started.md index 367e58284..e1bb1ce0c 100644 --- a/docs/versioned_docs/version-2.3/92-development/01-getting-started.md +++ b/docs/versioned_docs/version-2.6/92-development/01-getting-started.md @@ -1,12 +1,5 @@ # Getting started -## Core ideas - -- A (e.g. pipeline) configuration should never be [turing complete](https://en.wikipedia.org/wiki/Turing_completeness) (We have agents to exec things 🙂). -- If possible follow the [KISS principle](https://en.wikipedia.org/wiki/KISS_principle). -- What is used most should be default. -- Keep different topics separated, so you can write plugins, port new ideas ... more easily, see [Architecture](./05-architecture.md). - You can develop on your local computer by following the [steps below](#preparation-for-local-development) or you can start with a fully prepared online setup using [Gitpod](https://github.com/gitpod-io/gitpod) and [Gitea](https://github.com/go-gitea/gitea). ## Gitpod @@ -89,7 +82,7 @@ WOODPECKER_HEALTHCHECK=false ### Setup OAuth -Create an OAuth app for your forge as described in the [forges documentation](../30-administration/11-forges/10-overview.md). If you set `WOODPECKER_DEV_OAUTH_HOST=http://localhost:8000` you can use that address with the path as explained for the specific forge to login without the need for a public address. For example for GitHub you would use `http://localhost:8000/authorize` as authorization callback URL. +Create an OAuth app for your forge as described in the [forges documentation](../30-administration/11-forges/11-overview.md). If you set `WOODPECKER_DEV_OAUTH_HOST=http://localhost:8000` you can use that address with the path as explained for the specific forge to login without the need for a public address. For example for GitHub you would use `http://localhost:8000/authorize` as authorization callback URL. ## Developing with VS Code diff --git a/docs/versioned_docs/version-2.6/92-development/02-core-ideas.md b/docs/versioned_docs/version-2.6/92-development/02-core-ideas.md new file mode 100644 index 000000000..8e0d6e292 --- /dev/null +++ b/docs/versioned_docs/version-2.6/92-development/02-core-ideas.md @@ -0,0 +1,26 @@ +# Core ideas + +- A configuration (e.g. of a pipeline) should never be [turing complete](https://en.wikipedia.org/wiki/Turing_completeness) (We have agents to exec things 🙂). +- If possible, follow the [KISS principle](https://en.wikipedia.org/wiki/KISS_principle). +- What is used most often should be default. +- Keep different topics separated, so you can write plugins, port new ideas ... more easily, see [Architecture](./05-architecture.md). + +## Addons and extensions + +If you are wondering whether your contribution will be accepted to be merged in the Woodpecker core, or whether it's better to write an +[addon forge](../30-administration/11-forges/100-addon.md), [extension](../30-administration/100-external-configuration-api.md) or an +[external custom backend](../30-administration/22-backends/50-custom-backends.md), please check these points: + +- Is your change very specific to your setup and unlikely to be used by anyone else? +- Does your change violate the [guidelines](#guidelines)? + +Both should be false when you open a pull request to get your change into the core repository. + +### Guidelines + +#### Forges + +A new forge must support these features: + +- OAuth2 +- Webhooks diff --git a/docs/versioned_docs/version-2.3/92-development/03-ui.md b/docs/versioned_docs/version-2.6/92-development/03-ui.md similarity index 100% rename from docs/versioned_docs/version-2.3/92-development/03-ui.md rename to docs/versioned_docs/version-2.6/92-development/03-ui.md diff --git a/docs/versioned_docs/version-2.3/92-development/04-docs.md b/docs/versioned_docs/version-2.6/92-development/04-docs.md similarity index 100% rename from docs/versioned_docs/version-2.3/92-development/04-docs.md rename to docs/versioned_docs/version-2.6/92-development/04-docs.md diff --git a/docs/versioned_docs/version-2.3/92-development/05-architecture.md b/docs/versioned_docs/version-2.6/92-development/05-architecture.md similarity index 92% rename from docs/versioned_docs/version-2.3/92-development/05-architecture.md rename to docs/versioned_docs/version-2.6/92-development/05-architecture.md index 466ab2171..ccfd410f3 100644 --- a/docs/versioned_docs/version-2.3/92-development/05-architecture.md +++ b/docs/versioned_docs/version-2.6/92-development/05-architecture.md @@ -34,8 +34,8 @@ | `server/forge/**` | forge lib for server to connect and handle forge specific stuff | `shared`, `server/model` | | `server/router/**` | handle requests to REST API (and all middleware) and serve UI and WebUI config | `shared`, `../api`, `../model`, `../forge`, `../store`, `../web` | | `server/store/**` | handle database | `server/model` | -| `server/shared/**` | TODO: move and split [#974](https://github.com/woodpecker-ci/woodpecker/issues/974) | -| `server/web/**` | server SPA | +| `server/shared/**` | TODO: move and split [#974](https://github.com/woodpecker-ci/woodpecker/issues/974) | | +| `server/web/**` | server SPA | | - `../` = `server/` diff --git a/docs/versioned_docs/version-2.3/92-development/06-guides.md b/docs/versioned_docs/version-2.6/92-development/06-guides.md similarity index 92% rename from docs/versioned_docs/version-2.3/92-development/06-guides.md rename to docs/versioned_docs/version-2.6/92-development/06-guides.md index e8db28a53..c70a9ec93 100644 --- a/docs/versioned_docs/version-2.3/92-development/06-guides.md +++ b/docs/versioned_docs/version-2.6/92-development/06-guides.md @@ -3,7 +3,6 @@ ## ORM Woodpecker uses [Xorm](https://xorm.io/) as ORM for the database connection. -You can find its documentation at [gobook.io/read/gitea.com/xorm](https://gobook.io/read/gitea.com/xorm/manual-en-US/). ## Add a new migration diff --git a/docs/versioned_docs/version-2.3/92-development/07-translations.md b/docs/versioned_docs/version-2.6/92-development/07-translations.md similarity index 100% rename from docs/versioned_docs/version-2.3/92-development/07-translations.md rename to docs/versioned_docs/version-2.6/92-development/07-translations.md diff --git a/docs/versioned_docs/version-2.3/92-development/08-swagger.md b/docs/versioned_docs/version-2.6/92-development/08-swagger.md similarity index 97% rename from docs/versioned_docs/version-2.3/92-development/08-swagger.md rename to docs/versioned_docs/version-2.6/92-development/08-swagger.md index 92505a8fc..9a3775c41 100644 --- a/docs/versioned_docs/version-2.3/92-development/08-swagger.md +++ b/docs/versioned_docs/version-2.6/92-development/08-swagger.md @@ -46,7 +46,7 @@ These guidelines aim to have consistent wording in the swagger doc: - `@Param Authorization` is almost always present, there are just a few un-protected endpoints There are many examples in the `server/api` package, which you can use a blueprint. -More enhanced information you can find here +More enhanced information you can find here ### Manual code generation diff --git a/docs/versioned_docs/version-2.3/92-development/_category_.yaml b/docs/versioned_docs/version-2.6/92-development/_category_.yaml similarity index 100% rename from docs/versioned_docs/version-2.3/92-development/_category_.yaml rename to docs/versioned_docs/version-2.6/92-development/_category_.yaml diff --git a/docs/versioned_docs/version-2.3/92-development/ui-proxy.svg b/docs/versioned_docs/version-2.6/92-development/ui-proxy.svg similarity index 100% rename from docs/versioned_docs/version-2.3/92-development/ui-proxy.svg rename to docs/versioned_docs/version-2.6/92-development/ui-proxy.svg diff --git a/docs/versioned_docs/version-2.3/92-development/vscode-debug.png b/docs/versioned_docs/version-2.6/92-development/vscode-debug.png similarity index 100% rename from docs/versioned_docs/version-2.3/92-development/vscode-debug.png rename to docs/versioned_docs/version-2.6/92-development/vscode-debug.png diff --git a/docs/versioned_docs/version-2.3/92-development/vscode-run-test.png b/docs/versioned_docs/version-2.6/92-development/vscode-run-test.png similarity index 100% rename from docs/versioned_docs/version-2.3/92-development/vscode-run-test.png rename to docs/versioned_docs/version-2.6/92-development/vscode-run-test.png diff --git a/docs/versioned_docs/version-2.3/92-development/woodpecker-architecture.png b/docs/versioned_docs/version-2.6/92-development/woodpecker-architecture.png similarity index 100% rename from docs/versioned_docs/version-2.3/92-development/woodpecker-architecture.png rename to docs/versioned_docs/version-2.6/92-development/woodpecker-architecture.png diff --git a/docs/versioned_docs/version-2.3/woodpecker.png b/docs/versioned_docs/version-2.6/woodpecker.png similarity index 100% rename from docs/versioned_docs/version-2.3/woodpecker.png rename to docs/versioned_docs/version-2.6/woodpecker.png diff --git a/docs/versioned_sidebars/version-2.3-sidebars.json b/docs/versioned_sidebars/version-2.6-sidebars.json similarity index 100% rename from docs/versioned_sidebars/version-2.3-sidebars.json rename to docs/versioned_sidebars/version-2.6-sidebars.json diff --git a/docs/versions.json b/docs/versions.json index ebc024499..317e4f49d 100644 --- a/docs/versions.json +++ b/docs/versions.json @@ -1 +1 @@ -["2.5", "2.4", "2.3", "1.0"] +["2.6", "2.5", "2.4", "1.0"]