Skip to content

Commit 1cc5525

Browse files
committed
feat: init
feat: modify action.yml feat: modify action.yml test: test output test: test output test: test output test: test output test: test output test: test output test: test output test: test output test: test output test: test output test: test output test: test output test: test output feat: publish
1 parent ce4d174 commit 1cc5525

File tree

379 files changed

+79061
-1
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

379 files changed

+79061
-1
lines changed

.gitignore

+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
# Logs
2+
logs
3+
*.log
4+
npm-debug.log*
5+
yarn-debug.log*
6+
yarn-error.log*
7+
lerna-debug.log*
8+
9+
# Diagnostic reports (https://nodejs.org/api/report.html)
10+
report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
11+
12+
# Runtime data
13+
pids
14+
*.pid
15+
*.seed
16+
*.pid.lock
17+
18+
# Directory for instrumented libs generated by jscoverage/JSCover
19+
lib-cov
20+
21+
# Coverage directory used by tools like istanbul
22+
coverage
23+
*.lcov
24+
25+
# nyc test coverage
26+
.nyc_output
27+
28+
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
29+
.grunt
30+
31+
# Bower dependency directory (https://bower.io/)
32+
bower_components
33+
34+
# node-waf configuration
35+
.lock-wscript
36+
37+
# Compiled binary addons (https://nodejs.org/api/addons.html)
38+
build/Release
39+
40+
# Dependency directories
41+
jspm_packages/
42+
43+
# TypeScript v1 declaration files
44+
typings/
45+
46+
# TypeScript cache
47+
*.tsbuildinfo
48+
49+
# Optional npm cache directory
50+
.npm
51+
52+
# Optional eslint cache
53+
.eslintcache
54+
55+
# Microbundle cache
56+
.rpt2_cache/
57+
.rts2_cache_cjs/
58+
.rts2_cache_es/
59+
.rts2_cache_umd/
60+
61+
# Optional REPL history
62+
.node_repl_history
63+
64+
# Output of 'npm pack'
65+
*.tgz
66+
67+
# Yarn Integrity file
68+
.yarn-integrity
69+
70+
# dotenv environment variables file
71+
.env
72+
.env.test
73+
74+
# parcel-bundler cache (https://parceljs.org/)
75+
.cache
76+
77+
# Next.js build output
78+
.next
79+
80+
# Nuxt.js build / generate output
81+
.nuxt
82+
83+
# Gatsby files
84+
.cache/
85+
# Comment in the public line in if your project uses Gatsby and *not* Next.js
86+
# https://nextjs.org/blog/next-9-1#public-directory-support
87+
# public
88+
89+
# vuepress build output
90+
.vuepress/dist
91+
92+
# Serverless directories
93+
.serverless/
94+
95+
# FuseBox cache
96+
.fusebox/
97+
98+
# DynamoDB Local files
99+
.dynamodb/
100+
101+
# TernJS port file
102+
.tern-port

README.md

+40-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,41 @@
11
# assign-reviewer
2-
this action help to assign your coworkers as pull request reviewer if condition's matched.
2+
3+
github only support [codeowner](https://docs.github.com/en/repositories/managing-your-repositorys-settings-and-features/customizing-your-repository/about-code-owners) currently. You can add a group of users as owner of specific folders or files.
4+
5+
But often, there're some features which will cross lots of files, like "telemetry", "log", etc. For some teams or company, there always be so called "Feature Owner" to own specific feature. And their coworkers will make Pull Request to contribute. This action will help to assign "Feature Owner" as reviewer of your Pull Request.
6+
7+
# usage
8+
9+
```yml
10+
- name: assign reviewer
11+
uses: LongOddCode/assign-reviewer
12+
with:
13+
#github access token.
14+
token: ${{ secrets.GITHUB_TOKEN }}
15+
16+
# Github action doesn't support arrary rightnow. So use json array
17+
# as a workaround.
18+
# Required.
19+
reviewers: '["Johnny Silverhand", "Neo", "John Wick", "Keanu"]'
20+
21+
# Number to assign to reviewer.
22+
# set 0 if you want assign all of them.
23+
# Optional. Default 0.
24+
conscript: 0
25+
26+
# Which kind of script do you wanna use.
27+
# Optional. Default "bash" on Linux & Mac. PowerShell on Windows.
28+
script: bash
29+
30+
# Set this as true if condition matched.
31+
# Required.
32+
result: TELEMETRY_RESULT
33+
34+
# Your business logic.
35+
# Required.
36+
run: |
37+
line=`git diff -U0 ${{ github.base_ref }} | grep '^[+-]' | grep -Ev '^(--- a/|\+\+\+ b/)' | grep -i "telemetry" | wc -l`
38+
if [ $line -gt 0 ]; then
39+
echo '::set-output name=TELEMETRY_RESULT::true'
40+
fi
41+
```

action.yml

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
name: "assign-reviewer"
2+
description: "this action help to assign your coworkers as pull request reviewer if condition's matched."
3+
4+
branding:
5+
icon: "tag"
6+
color: "blue"
7+
8+
inputs:
9+
token:
10+
description: github access token.
11+
required: true
12+
reviewers:
13+
description: reviewers candidates.
14+
required: true
15+
conscript:
16+
description: numerber to assign.
17+
required: false
18+
default: "0"
19+
script:
20+
description: Which kind of script do you wanna use.
21+
required: false
22+
default: ""
23+
result:
24+
description: Set this as true if condition matched.
25+
required: true
26+
run:
27+
description: Your business logic.
28+
required: true
29+
30+
runs:
31+
using: "node12"
32+
main: "index.js"

index.js

+156
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,156 @@
1+
const core = require("@actions/core");
2+
const github = require("@actions/github");
3+
const fs = require("fs-extra");
4+
const { exit } = require("process");
5+
const os = require("os");
6+
const path = require("path");
7+
const util = require("util");
8+
const execFile = util.promisify(require("child_process").execFile);
9+
10+
function paramValidation(param) {
11+
const value = core.getInput(param);
12+
if (value === "") {
13+
core.setFailed(`missing ${param}`);
14+
exit(1);
15+
}
16+
}
17+
18+
paramValidation("reviewers");
19+
paramValidation("token");
20+
paramValidation("result");
21+
paramValidation("run");
22+
23+
const octokit = github.getOctokit(core.getInput("token"));
24+
25+
/**
26+
* reviewers are json array.
27+
*/
28+
const reviewersString = core.getInput("reviewers");
29+
let reviewers = [];
30+
try {
31+
reviewers = JSON.parse(reviewersString);
32+
} catch (err) {
33+
core.setFailed("invalid reviewers");
34+
exit(1);
35+
}
36+
37+
/**
38+
* lan could be empty.
39+
* Set bash as default on Linux & Mac.
40+
* Set PowerShell as default on Windows.
41+
*/
42+
const lan = core.getInput("script");
43+
if (lan === "") {
44+
switch (os.type()) {
45+
case "Linux":
46+
lan = "bash";
47+
break;
48+
case "Darwin":
49+
lan = "bash";
50+
break;
51+
case "Windows_NT":
52+
lan = "pwsh";
53+
break;
54+
default:
55+
lan = "bash";
56+
}
57+
}
58+
59+
/**
60+
* conscript could be empty. Set reviewers.length as default.
61+
*/
62+
const conscriptString = core.getInput("conscript");
63+
let conscript = reviewers.length;
64+
try {
65+
conscript = parseInt(conscriptString, 10);
66+
} catch (err) {
67+
core.setFailed("invalid conscript");
68+
exit(1);
69+
}
70+
71+
const result = core.getInput("result");
72+
const code = core.getMultilineInput("run").join(os.EOL);
73+
74+
/**
75+
* this func will help to add feature owners to PR reviewer.
76+
*/
77+
function addReviewers() {
78+
if (conscript <= 0) {
79+
conscript = reviewers.length;
80+
}
81+
octokit
82+
.request(
83+
"POST /repos/{owner}/{repo}/pulls/{pull_number}/requested_reviewers",
84+
{
85+
owner: github.context.repo.owner,
86+
repo: github.context.repo.repo,
87+
pull_number: github.context.payload.number,
88+
reviewers: (reviewers = reviewers.slice(0, conscript)),
89+
}
90+
)
91+
.then(() => {
92+
console.log("Add reviewers successfully");
93+
})
94+
.catch((err) => {
95+
console.error(err);
96+
exit(1);
97+
});
98+
}
99+
100+
/**
101+
* out will flow github action pattern, like '::set-output name=TELEMETRY_RESULT::true'.
102+
* return result's value if included.
103+
*/
104+
function matchResult(out, result) {
105+
if (out.includes(`::set-output name=${result}::true`)) {
106+
return true;
107+
} else {
108+
return false;
109+
}
110+
}
111+
112+
/**
113+
* this func will execute user mannualed script.
114+
*/
115+
async function run(lan, code, result) {
116+
let suffix = "";
117+
switch (lan) {
118+
case "bash":
119+
code = `#!/bin/bash${os.EOL}` + code;
120+
suffix = "sh";
121+
break;
122+
case "sh":
123+
code = `#!/bin/sh${os.EOL}` + code;
124+
suffix = "sh";
125+
break;
126+
case "pwsh":
127+
suffix = "ps1";
128+
break;
129+
default:
130+
console.error("unsupported script");
131+
exit(1);
132+
}
133+
134+
let dirPath = path.join(os.homedir(), "action");
135+
try {
136+
dirPath = await fs.mkdtemp(dirPath);
137+
const filePath = path.join(dirPath, `assign-reviewer.${suffix}`);
138+
139+
await fs.writeFile(filePath, code, {
140+
encoding: "utf8",
141+
mode: 0o777,
142+
});
143+
144+
const { stdout } = await execFile(filePath);
145+
if (matchResult(stdout, result)) {
146+
addReviewers();
147+
}
148+
console.log(stdout);
149+
} catch (err) {
150+
console.error(err);
151+
} finally {
152+
fs.rmdirSync(dirPath, { recursive: true });
153+
}
154+
}
155+
156+
run(lan, code, result).then();

node_modules/.bin/node

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

node_modules/@actions/core/LICENSE.md

+9
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)