Implement actions (#21937)
Close #13539.
Co-authored by: @lunny @appleboy @fuxiaohei and others.
Related projects:
- https://gitea.com/gitea/actions-proto-def
- https://gitea.com/gitea/actions-proto-go
- https://gitea.com/gitea/act
- https://gitea.com/gitea/act_runner
### Summary
The target of this PR is to bring a basic implementation of "Actions",
an internal CI/CD system of Gitea. That means even though it has been
merged, the state of the feature is **EXPERIMENTAL**, and please note
that:
- It is disabled by default;
- It shouldn't be used in a production environment currently;
- It shouldn't be used in a public Gitea instance currently;
- Breaking changes may be made before it's stable.
**Please comment on #13539 if you have any different product design
ideas**, all decisions reached there will be adopted here. But in this
PR, we don't talk about **naming, feature-creep or alternatives**.
### ⚠️ Breaking
`gitea-actions` will become a reserved user name. If a user with the
name already exists in the database, it is recommended to rename it.
### Some important reviews
- What is `DEFAULT_ACTIONS_URL` in `app.ini` for?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1055954954
- Why the api for runners is not under the normal `/api/v1` prefix?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1061173592
- Why DBFS?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1061301178
- Why ignore events triggered by `gitea-actions` bot?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1063254103
- Why there's no permission control for actions?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1090229868
### What it looks like
<details>
#### Manage runners
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205870657-c72f590e-2e08-4cd4-be7f-2e0abb299bbf.png">
#### List runs
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205872794-50fde990-2b45-48c1-a178-908e4ec5b627.png">
#### View logs
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205872501-9b7b9000-9542-4991-8f55-18ccdada77c3.png">
</details>
### How to try it
<details>
#### 1. Start Gitea
Clone this branch and [install from
source](https://docs.gitea.io/en-us/install-from-source).
Add additional configurations in `app.ini` to enable Actions:
```ini
[actions]
ENABLED = true
```
Start it.
If all is well, you'll see the management page of runners:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205877365-8e30a780-9b10-4154-b3e8-ee6c3cb35a59.png">
#### 2. Start runner
Clone the [act_runner](https://gitea.com/gitea/act_runner), and follow
the
[README](https://gitea.com/gitea/act_runner/src/branch/main/README.md)
to start it.
If all is well, you'll see a new runner has been added:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205878000-216f5937-e696-470d-b66c-8473987d91c3.png">
#### 3. Enable actions for a repo
Create a new repo or open an existing one, check the `Actions` checkbox
in settings and submit.
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205879705-53e09208-73c0-4b3e-a123-2dcf9aba4b9c.png">
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205879383-23f3d08f-1a85-41dd-a8b3-54e2ee6453e8.png">
If all is well, you'll see a new tab "Actions":
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205881648-a8072d8c-5803-4d76-b8a8-9b2fb49516c1.png">
#### 4. Upload workflow files
Upload some workflow files to `.gitea/workflows/xxx.yaml`, you can
follow the [quickstart](https://docs.github.com/en/actions/quickstart)
of GitHub Actions. Yes, Gitea Actions is compatible with GitHub Actions
in most cases, you can use the same demo:
```yaml
name: GitHub Actions Demo
run-name: ${{ github.actor }} is testing out GitHub Actions 🚀
on: [push]
jobs:
Explore-GitHub-Actions:
runs-on: ubuntu-latest
steps:
- run: echo "🎉 The job was automatically triggered by a ${{ github.event_name }} event."
- run: echo "🐧 This job is now running on a ${{ runner.os }} server hosted by GitHub!"
- run: echo "🔎 The name of your branch is ${{ github.ref }} and your repository is ${{ github.repository }}."
- name: Check out repository code
uses: actions/checkout@v3
- run: echo "💡 The ${{ github.repository }} repository has been cloned to the runner."
- run: echo "🖥️ The workflow is now ready to test your code on the runner."
- name: List files in the repository
run: |
ls ${{ github.workspace }}
- run: echo "🍏 This job's status is ${{ job.status }}."
```
If all is well, you'll see a new run in `Actions` tab:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205884473-79a874bc-171b-4aaf-acd5-0241a45c3b53.png">
#### 5. Check the logs of jobs
Click a run and you'll see the logs:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205884800-994b0374-67f7-48ff-be9a-4c53f3141547.png">
#### 6. Go on
You can try more examples in [the
documents](https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions)
of GitHub Actions, then you might find a lot of bugs.
Come on, PRs are welcome.
</details>
See also: [Feature Preview: Gitea
Actions](https://blog.gitea.io/2022/12/feature-preview-gitea-actions/)
---------
Co-authored-by: a1012112796 <1012112796@qq.com>
Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
Co-authored-by: delvh <dev.lh@web.de>
Co-authored-by: ChristopherHX <christopher.homberger@web.de>
Co-authored-by: John Olheiser <john.olheiser@gmail.com>
2023-01-31 09:45:19 +08:00
// Copyright 2022 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package actions
import (
"context"
"crypto/subtle"
"fmt"
"time"
auth_model "code.gitea.io/gitea/models/auth"
"code.gitea.io/gitea/models/db"
"code.gitea.io/gitea/modules/container"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/timeutil"
"code.gitea.io/gitea/modules/util"
runnerv1 "code.gitea.io/actions-proto-go/runner/v1"
lru "github.com/hashicorp/golang-lru"
"github.com/nektos/act/pkg/jobparser"
"google.golang.org/protobuf/types/known/timestamppb"
"xorm.io/builder"
)
// ActionTask represents a distribution of job
type ActionTask struct {
ID int64
JobID int64
Job * ActionRunJob ` xorm:"-" `
Steps [ ] * ActionTaskStep ` xorm:"-" `
Attempt int64
RunnerID int64 ` xorm:"index" `
Status Status ` xorm:"index" `
Started timeutil . TimeStamp ` xorm:"index" `
Stopped timeutil . TimeStamp
RepoID int64 ` xorm:"index" `
OwnerID int64 ` xorm:"index" `
CommitSHA string ` xorm:"index" `
IsForkPullRequest bool
Token string ` xorm:"-" `
TokenHash string ` xorm:"UNIQUE" ` // sha256 of token
TokenSalt string
TokenLastEight string ` xorm:"index token_last_eight" `
LogFilename string // file name of log
LogInStorage bool // read log from database or from storage
LogLength int64 // lines count
LogSize int64 // blob size
LogIndexes LogIndexes ` xorm:"LONGBLOB" ` // line number to offset
LogExpired bool // files that are too old will be deleted
Created timeutil . TimeStamp ` xorm:"created" `
Updated timeutil . TimeStamp ` xorm:"updated index" `
}
var successfulTokenTaskCache * lru . Cache
func init ( ) {
db . RegisterModel ( new ( ActionTask ) , func ( ) error {
if setting . SuccessfulTokensCacheSize > 0 {
var err error
successfulTokenTaskCache , err = lru . New ( setting . SuccessfulTokensCacheSize )
if err != nil {
return fmt . Errorf ( "unable to allocate Task cache: %v" , err )
}
} else {
successfulTokenTaskCache = nil
}
return nil
} )
}
func ( task * ActionTask ) Duration ( ) time . Duration {
return calculateDuration ( task . Started , task . Stopped , task . Status )
}
func ( task * ActionTask ) IsStopped ( ) bool {
return task . Stopped > 0
}
func ( task * ActionTask ) GetRunLink ( ) string {
if task . Job == nil || task . Job . Run == nil {
return ""
}
return task . Job . Run . Link ( )
}
func ( task * ActionTask ) GetCommitLink ( ) string {
if task . Job == nil || task . Job . Run == nil || task . Job . Run . Repo == nil {
return ""
}
return task . Job . Run . Repo . CommitLink ( task . CommitSHA )
}
func ( task * ActionTask ) GetRepoName ( ) string {
if task . Job == nil || task . Job . Run == nil || task . Job . Run . Repo == nil {
return ""
}
return task . Job . Run . Repo . FullName ( )
}
func ( task * ActionTask ) GetRepoLink ( ) string {
if task . Job == nil || task . Job . Run == nil || task . Job . Run . Repo == nil {
return ""
}
return task . Job . Run . Repo . Link ( )
}
func ( task * ActionTask ) LoadJob ( ctx context . Context ) error {
if task . Job == nil {
job , err := GetRunJobByID ( ctx , task . JobID )
if err != nil {
return err
}
task . Job = job
}
return nil
}
// LoadAttributes load Job Steps if not loaded
func ( task * ActionTask ) LoadAttributes ( ctx context . Context ) error {
if task == nil {
return nil
}
if err := task . LoadJob ( ctx ) ; err != nil {
return err
}
if err := task . Job . LoadAttributes ( ctx ) ; err != nil {
return err
}
if task . Steps == nil { // be careful, an empty slice (not nil) also means loaded
steps , err := GetTaskStepsByTaskID ( ctx , task . ID )
if err != nil {
return err
}
task . Steps = steps
}
return nil
}
func ( task * ActionTask ) GenerateToken ( ) ( err error ) {
task . Token , task . TokenSalt , task . TokenHash , task . TokenLastEight , err = generateSaltedToken ( )
return err
}
func GetTaskByID ( ctx context . Context , id int64 ) ( * ActionTask , error ) {
var task ActionTask
has , err := db . GetEngine ( ctx ) . Where ( "id=?" , id ) . Get ( & task )
if err != nil {
return nil , err
} else if ! has {
return nil , fmt . Errorf ( "task with id %d: %w" , id , util . ErrNotExist )
}
return & task , nil
}
func GetRunningTaskByToken ( ctx context . Context , token string ) ( * ActionTask , error ) {
errNotExist := fmt . Errorf ( "task with token %q: %w" , token , util . ErrNotExist )
if token == "" {
return nil , errNotExist
}
// A token is defined as being SHA1 sum these are 40 hexadecimal bytes long
if len ( token ) != 40 {
return nil , errNotExist
}
for _ , x := range [ ] byte ( token ) {
if x < '0' || ( x > '9' && x < 'a' ) || x > 'f' {
return nil , errNotExist
}
}
lastEight := token [ len ( token ) - 8 : ]
if id := getTaskIDFromCache ( token ) ; id > 0 {
task := & ActionTask {
TokenLastEight : lastEight ,
}
// Re-get the task from the db in case it has been deleted in the intervening period
has , err := db . GetEngine ( ctx ) . ID ( id ) . Get ( task )
if err != nil {
return nil , err
}
if has {
return task , nil
}
successfulTokenTaskCache . Remove ( token )
}
var tasks [ ] * ActionTask
err := db . GetEngine ( ctx ) . Where ( "token_last_eight = ? AND status = ?" , lastEight , StatusRunning ) . Find ( & tasks )
if err != nil {
return nil , err
} else if len ( tasks ) == 0 {
return nil , errNotExist
}
for _ , t := range tasks {
tempHash := auth_model . HashToken ( token , t . TokenSalt )
if subtle . ConstantTimeCompare ( [ ] byte ( t . TokenHash ) , [ ] byte ( tempHash ) ) == 1 {
if successfulTokenTaskCache != nil {
successfulTokenTaskCache . Add ( token , t . ID )
}
return t , nil
}
}
return nil , errNotExist
}
func CreateTaskForRunner ( ctx context . Context , runner * ActionRunner ) ( * ActionTask , bool , error ) {
dbCtx , commiter , err := db . TxContext ( ctx )
if err != nil {
return nil , false , err
}
defer commiter . Close ( )
ctx = dbCtx . WithContext ( ctx )
e := db . GetEngine ( ctx )
jobCond := builder . NewCond ( )
if runner . RepoID != 0 {
jobCond = builder . Eq { "repo_id" : runner . RepoID }
} else if runner . OwnerID != 0 {
jobCond = builder . In ( "repo_id" , builder . Select ( "id" ) . From ( "repository" ) . Where ( builder . Eq { "owner_id" : runner . OwnerID } ) )
}
if jobCond . IsValid ( ) {
jobCond = builder . In ( "run_id" , builder . Select ( "id" ) . From ( "action_run" ) . Where ( jobCond ) )
}
var jobs [ ] * ActionRunJob
if err := e . Where ( "task_id=? AND status=?" , 0 , StatusWaiting ) . And ( jobCond ) . Asc ( "id" ) . Find ( & jobs ) ; err != nil {
return nil , false , err
}
// TODO: a more efficient way to filter labels
var job * ActionRunJob
labels := runner . AgentLabels
labels = append ( labels , runner . CustomLabels ... )
log . Trace ( "runner labels: %v" , labels )
for _ , v := range jobs {
if isSubset ( labels , v . RunsOn ) {
job = v
break
}
}
if job == nil {
return nil , false , nil
}
if err := job . LoadAttributes ( ctx ) ; err != nil {
return nil , false , err
}
now := timeutil . TimeStampNow ( )
job . Attempt ++
job . Started = now
job . Status = StatusRunning
task := & ActionTask {
JobID : job . ID ,
Attempt : job . Attempt ,
RunnerID : runner . ID ,
Started : now ,
Status : StatusRunning ,
RepoID : job . RepoID ,
OwnerID : job . OwnerID ,
CommitSHA : job . CommitSHA ,
IsForkPullRequest : job . IsForkPullRequest ,
}
if err := task . GenerateToken ( ) ; err != nil {
return nil , false , err
}
var workflowJob * jobparser . Job
if gots , err := jobparser . Parse ( job . WorkflowPayload ) ; err != nil {
return nil , false , fmt . Errorf ( "parse workflow of job %d: %w" , job . ID , err )
} else if len ( gots ) != 1 {
return nil , false , fmt . Errorf ( "workflow of job %d: not signle workflow" , job . ID )
} else {
_ , workflowJob = gots [ 0 ] . Job ( )
}
if _ , err := e . Insert ( task ) ; err != nil {
return nil , false , err
}
task . LogFilename = logFileName ( job . Run . Repo . FullName ( ) , task . ID )
if _ , err := e . ID ( task . ID ) . Cols ( "log_filename" ) . Update ( task ) ; err != nil {
return nil , false , err
}
if len ( workflowJob . Steps ) > 0 {
steps := make ( [ ] * ActionTaskStep , len ( workflowJob . Steps ) )
for i , v := range workflowJob . Steps {
2023-02-28 18:20:36 +08:00
name , _ := util . SplitStringAtByteN ( v . String ( ) , 255 )
Implement actions (#21937)
Close #13539.
Co-authored by: @lunny @appleboy @fuxiaohei and others.
Related projects:
- https://gitea.com/gitea/actions-proto-def
- https://gitea.com/gitea/actions-proto-go
- https://gitea.com/gitea/act
- https://gitea.com/gitea/act_runner
### Summary
The target of this PR is to bring a basic implementation of "Actions",
an internal CI/CD system of Gitea. That means even though it has been
merged, the state of the feature is **EXPERIMENTAL**, and please note
that:
- It is disabled by default;
- It shouldn't be used in a production environment currently;
- It shouldn't be used in a public Gitea instance currently;
- Breaking changes may be made before it's stable.
**Please comment on #13539 if you have any different product design
ideas**, all decisions reached there will be adopted here. But in this
PR, we don't talk about **naming, feature-creep or alternatives**.
### ⚠️ Breaking
`gitea-actions` will become a reserved user name. If a user with the
name already exists in the database, it is recommended to rename it.
### Some important reviews
- What is `DEFAULT_ACTIONS_URL` in `app.ini` for?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1055954954
- Why the api for runners is not under the normal `/api/v1` prefix?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1061173592
- Why DBFS?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1061301178
- Why ignore events triggered by `gitea-actions` bot?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1063254103
- Why there's no permission control for actions?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1090229868
### What it looks like
<details>
#### Manage runners
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205870657-c72f590e-2e08-4cd4-be7f-2e0abb299bbf.png">
#### List runs
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205872794-50fde990-2b45-48c1-a178-908e4ec5b627.png">
#### View logs
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205872501-9b7b9000-9542-4991-8f55-18ccdada77c3.png">
</details>
### How to try it
<details>
#### 1. Start Gitea
Clone this branch and [install from
source](https://docs.gitea.io/en-us/install-from-source).
Add additional configurations in `app.ini` to enable Actions:
```ini
[actions]
ENABLED = true
```
Start it.
If all is well, you'll see the management page of runners:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205877365-8e30a780-9b10-4154-b3e8-ee6c3cb35a59.png">
#### 2. Start runner
Clone the [act_runner](https://gitea.com/gitea/act_runner), and follow
the
[README](https://gitea.com/gitea/act_runner/src/branch/main/README.md)
to start it.
If all is well, you'll see a new runner has been added:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205878000-216f5937-e696-470d-b66c-8473987d91c3.png">
#### 3. Enable actions for a repo
Create a new repo or open an existing one, check the `Actions` checkbox
in settings and submit.
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205879705-53e09208-73c0-4b3e-a123-2dcf9aba4b9c.png">
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205879383-23f3d08f-1a85-41dd-a8b3-54e2ee6453e8.png">
If all is well, you'll see a new tab "Actions":
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205881648-a8072d8c-5803-4d76-b8a8-9b2fb49516c1.png">
#### 4. Upload workflow files
Upload some workflow files to `.gitea/workflows/xxx.yaml`, you can
follow the [quickstart](https://docs.github.com/en/actions/quickstart)
of GitHub Actions. Yes, Gitea Actions is compatible with GitHub Actions
in most cases, you can use the same demo:
```yaml
name: GitHub Actions Demo
run-name: ${{ github.actor }} is testing out GitHub Actions 🚀
on: [push]
jobs:
Explore-GitHub-Actions:
runs-on: ubuntu-latest
steps:
- run: echo "🎉 The job was automatically triggered by a ${{ github.event_name }} event."
- run: echo "🐧 This job is now running on a ${{ runner.os }} server hosted by GitHub!"
- run: echo "🔎 The name of your branch is ${{ github.ref }} and your repository is ${{ github.repository }}."
- name: Check out repository code
uses: actions/checkout@v3
- run: echo "💡 The ${{ github.repository }} repository has been cloned to the runner."
- run: echo "🖥️ The workflow is now ready to test your code on the runner."
- name: List files in the repository
run: |
ls ${{ github.workspace }}
- run: echo "🍏 This job's status is ${{ job.status }}."
```
If all is well, you'll see a new run in `Actions` tab:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205884473-79a874bc-171b-4aaf-acd5-0241a45c3b53.png">
#### 5. Check the logs of jobs
Click a run and you'll see the logs:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205884800-994b0374-67f7-48ff-be9a-4c53f3141547.png">
#### 6. Go on
You can try more examples in [the
documents](https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions)
of GitHub Actions, then you might find a lot of bugs.
Come on, PRs are welcome.
</details>
See also: [Feature Preview: Gitea
Actions](https://blog.gitea.io/2022/12/feature-preview-gitea-actions/)
---------
Co-authored-by: a1012112796 <1012112796@qq.com>
Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
Co-authored-by: delvh <dev.lh@web.de>
Co-authored-by: ChristopherHX <christopher.homberger@web.de>
Co-authored-by: John Olheiser <john.olheiser@gmail.com>
2023-01-31 09:45:19 +08:00
steps [ i ] = & ActionTaskStep {
2023-02-28 18:20:36 +08:00
Name : name ,
Implement actions (#21937)
Close #13539.
Co-authored by: @lunny @appleboy @fuxiaohei and others.
Related projects:
- https://gitea.com/gitea/actions-proto-def
- https://gitea.com/gitea/actions-proto-go
- https://gitea.com/gitea/act
- https://gitea.com/gitea/act_runner
### Summary
The target of this PR is to bring a basic implementation of "Actions",
an internal CI/CD system of Gitea. That means even though it has been
merged, the state of the feature is **EXPERIMENTAL**, and please note
that:
- It is disabled by default;
- It shouldn't be used in a production environment currently;
- It shouldn't be used in a public Gitea instance currently;
- Breaking changes may be made before it's stable.
**Please comment on #13539 if you have any different product design
ideas**, all decisions reached there will be adopted here. But in this
PR, we don't talk about **naming, feature-creep or alternatives**.
### ⚠️ Breaking
`gitea-actions` will become a reserved user name. If a user with the
name already exists in the database, it is recommended to rename it.
### Some important reviews
- What is `DEFAULT_ACTIONS_URL` in `app.ini` for?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1055954954
- Why the api for runners is not under the normal `/api/v1` prefix?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1061173592
- Why DBFS?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1061301178
- Why ignore events triggered by `gitea-actions` bot?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1063254103
- Why there's no permission control for actions?
- https://github.com/go-gitea/gitea/pull/21937#discussion_r1090229868
### What it looks like
<details>
#### Manage runners
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205870657-c72f590e-2e08-4cd4-be7f-2e0abb299bbf.png">
#### List runs
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205872794-50fde990-2b45-48c1-a178-908e4ec5b627.png">
#### View logs
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205872501-9b7b9000-9542-4991-8f55-18ccdada77c3.png">
</details>
### How to try it
<details>
#### 1. Start Gitea
Clone this branch and [install from
source](https://docs.gitea.io/en-us/install-from-source).
Add additional configurations in `app.ini` to enable Actions:
```ini
[actions]
ENABLED = true
```
Start it.
If all is well, you'll see the management page of runners:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205877365-8e30a780-9b10-4154-b3e8-ee6c3cb35a59.png">
#### 2. Start runner
Clone the [act_runner](https://gitea.com/gitea/act_runner), and follow
the
[README](https://gitea.com/gitea/act_runner/src/branch/main/README.md)
to start it.
If all is well, you'll see a new runner has been added:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205878000-216f5937-e696-470d-b66c-8473987d91c3.png">
#### 3. Enable actions for a repo
Create a new repo or open an existing one, check the `Actions` checkbox
in settings and submit.
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205879705-53e09208-73c0-4b3e-a123-2dcf9aba4b9c.png">
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205879383-23f3d08f-1a85-41dd-a8b3-54e2ee6453e8.png">
If all is well, you'll see a new tab "Actions":
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205881648-a8072d8c-5803-4d76-b8a8-9b2fb49516c1.png">
#### 4. Upload workflow files
Upload some workflow files to `.gitea/workflows/xxx.yaml`, you can
follow the [quickstart](https://docs.github.com/en/actions/quickstart)
of GitHub Actions. Yes, Gitea Actions is compatible with GitHub Actions
in most cases, you can use the same demo:
```yaml
name: GitHub Actions Demo
run-name: ${{ github.actor }} is testing out GitHub Actions 🚀
on: [push]
jobs:
Explore-GitHub-Actions:
runs-on: ubuntu-latest
steps:
- run: echo "🎉 The job was automatically triggered by a ${{ github.event_name }} event."
- run: echo "🐧 This job is now running on a ${{ runner.os }} server hosted by GitHub!"
- run: echo "🔎 The name of your branch is ${{ github.ref }} and your repository is ${{ github.repository }}."
- name: Check out repository code
uses: actions/checkout@v3
- run: echo "💡 The ${{ github.repository }} repository has been cloned to the runner."
- run: echo "🖥️ The workflow is now ready to test your code on the runner."
- name: List files in the repository
run: |
ls ${{ github.workspace }}
- run: echo "🍏 This job's status is ${{ job.status }}."
```
If all is well, you'll see a new run in `Actions` tab:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205884473-79a874bc-171b-4aaf-acd5-0241a45c3b53.png">
#### 5. Check the logs of jobs
Click a run and you'll see the logs:
<img width="1792" alt="image"
src="https://user-images.githubusercontent.com/9418365/205884800-994b0374-67f7-48ff-be9a-4c53f3141547.png">
#### 6. Go on
You can try more examples in [the
documents](https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions)
of GitHub Actions, then you might find a lot of bugs.
Come on, PRs are welcome.
</details>
See also: [Feature Preview: Gitea
Actions](https://blog.gitea.io/2022/12/feature-preview-gitea-actions/)
---------
Co-authored-by: a1012112796 <1012112796@qq.com>
Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
Co-authored-by: delvh <dev.lh@web.de>
Co-authored-by: ChristopherHX <christopher.homberger@web.de>
Co-authored-by: John Olheiser <john.olheiser@gmail.com>
2023-01-31 09:45:19 +08:00
TaskID : task . ID ,
Index : int64 ( i ) ,
RepoID : task . RepoID ,
Status : StatusWaiting ,
}
}
if _ , err := e . Insert ( steps ) ; err != nil {
return nil , false , err
}
task . Steps = steps
}
job . TaskID = task . ID
if n , err := UpdateRunJob ( ctx , job , builder . Eq { "task_id" : 0 } ) ; err != nil {
return nil , false , err
} else if n != 1 {
return nil , false , nil
}
if job . Run . Status . IsWaiting ( ) {
job . Run . Status = StatusRunning
job . Run . Started = now
if err := UpdateRun ( ctx , job . Run , "status" , "started" ) ; err != nil {
return nil , false , err
}
}
task . Job = job
if err := commiter . Commit ( ) ; err != nil {
return nil , false , err
}
return task , true , nil
}
func UpdateTask ( ctx context . Context , task * ActionTask , cols ... string ) error {
sess := db . GetEngine ( ctx ) . ID ( task . ID )
if len ( cols ) > 0 {
sess . Cols ( cols ... )
}
_ , err := sess . Update ( task )
return err
}
func UpdateTaskByState ( ctx context . Context , state * runnerv1 . TaskState ) ( * ActionTask , error ) {
stepStates := map [ int64 ] * runnerv1 . StepState { }
for _ , v := range state . Steps {
stepStates [ v . Id ] = v
}
ctx , commiter , err := db . TxContext ( ctx )
if err != nil {
return nil , err
}
defer commiter . Close ( )
e := db . GetEngine ( ctx )
task := & ActionTask { }
if has , err := e . ID ( state . Id ) . Get ( task ) ; err != nil {
return nil , err
} else if ! has {
return nil , util . ErrNotExist
}
if state . Result != runnerv1 . Result_RESULT_UNSPECIFIED {
task . Status = Status ( state . Result )
task . Stopped = timeutil . TimeStamp ( state . StoppedAt . AsTime ( ) . Unix ( ) )
if _ , err := UpdateRunJob ( ctx , & ActionRunJob {
ID : task . JobID ,
Status : task . Status ,
Stopped : task . Stopped ,
} , nil ) ; err != nil {
return nil , err
}
}
if _ , err := e . ID ( task . ID ) . Update ( task ) ; err != nil {
return nil , err
}
if err := task . LoadAttributes ( ctx ) ; err != nil {
return nil , err
}
for _ , step := range task . Steps {
var result runnerv1 . Result
if v , ok := stepStates [ step . Index ] ; ok {
result = v . Result
step . LogIndex = v . LogIndex
step . LogLength = v . LogLength
step . Started = convertTimestamp ( v . StartedAt )
step . Stopped = convertTimestamp ( v . StoppedAt )
}
if result != runnerv1 . Result_RESULT_UNSPECIFIED {
step . Status = Status ( result )
} else if step . Started != 0 {
step . Status = StatusRunning
}
if _ , err := e . ID ( step . ID ) . Update ( step ) ; err != nil {
return nil , err
}
}
if err := commiter . Commit ( ) ; err != nil {
return nil , err
}
return task , nil
}
func StopTask ( ctx context . Context , taskID int64 , status Status ) error {
if ! status . IsDone ( ) {
return fmt . Errorf ( "cannot stop task with status %v" , status )
}
e := db . GetEngine ( ctx )
task := & ActionTask { }
if has , err := e . ID ( taskID ) . Get ( task ) ; err != nil {
return err
} else if ! has {
return util . ErrNotExist
}
if task . Status . IsDone ( ) {
return nil
}
now := timeutil . TimeStampNow ( )
task . Status = status
task . Stopped = now
if _ , err := UpdateRunJob ( ctx , & ActionRunJob {
ID : task . JobID ,
Status : task . Status ,
Stopped : task . Stopped ,
} , nil ) ; err != nil {
return err
}
if _ , err := e . ID ( task . ID ) . Update ( task ) ; err != nil {
return err
}
if err := task . LoadAttributes ( ctx ) ; err != nil {
return err
}
for _ , step := range task . Steps {
if ! step . Status . IsDone ( ) {
step . Status = status
if step . Started == 0 {
step . Started = now
}
step . Stopped = now
}
if _ , err := e . ID ( step . ID ) . Update ( step ) ; err != nil {
return err
}
}
return nil
}
func isSubset ( set , subset [ ] string ) bool {
m := make ( container . Set [ string ] , len ( set ) )
for _ , v := range set {
m . Add ( v )
}
for _ , v := range subset {
if ! m . Contains ( v ) {
return false
}
}
return true
}
func convertTimestamp ( timestamp * timestamppb . Timestamp ) timeutil . TimeStamp {
if timestamp . GetSeconds ( ) == 0 && timestamp . GetNanos ( ) == 0 {
return timeutil . TimeStamp ( 0 )
}
return timeutil . TimeStamp ( timestamp . AsTime ( ) . Unix ( ) )
}
func logFileName ( repoFullName string , taskID int64 ) string {
return fmt . Sprintf ( "%s/%02x/%d.log" , repoFullName , taskID % 256 , taskID )
}
func getTaskIDFromCache ( token string ) int64 {
if successfulTokenTaskCache == nil {
return 0
}
tInterface , ok := successfulTokenTaskCache . Get ( token )
if ! ok {
return 0
}
t , ok := tInterface . ( int64 )
if ! ok {
return 0
}
return t
}