Extend awards action (#1937)
* remove debug job, restrict create-pull-request to only awards.txt, add documentation * make create-pull-request use a custom branch, and filter that out, so PRs generated by action don't invoke action again
This commit is contained in:
		
							parent
							
								
									8a764f0f75
								
							
						
					
					
						commit
						27104302d5
					
				
							
								
								
									
										32
									
								
								.github/workflows/extend-awards.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								.github/workflows/extend-awards.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@ -0,0 +1,32 @@
 | 
			
		||||
name: extend-awards
 | 
			
		||||
run-name: Extending awards
 | 
			
		||||
on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
    types: [ closed ]
 | 
			
		||||
    branches:
 | 
			
		||||
      - master
 | 
			
		||||
jobs:
 | 
			
		||||
  if_merged:
 | 
			
		||||
    if: |
 | 
			
		||||
      github.event_name == 'pull_request' &&
 | 
			
		||||
      github.event.action == 'closed' &&
 | 
			
		||||
      github.event.pull_request.merged == true &&
 | 
			
		||||
      github.event.pull_request.head.ref != 'extend-awards/patch'
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v4
 | 
			
		||||
      - uses: actions/setup-python@v5
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: '3.13' 
 | 
			
		||||
      - run: pip install requests
 | 
			
		||||
      - run: python extend-awards.py
 | 
			
		||||
        env:
 | 
			
		||||
          GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          GITHUB_CONTEXT: ${{ toJson(github) }}
 | 
			
		||||
      - uses: peter-evans/create-pull-request@v7
 | 
			
		||||
        with:
 | 
			
		||||
          add-paths: awards.csv
 | 
			
		||||
          branch: extend-awards/patch
 | 
			
		||||
          commit-message: Extending awards.csv
 | 
			
		||||
          title: Extending awards.csv
 | 
			
		||||
          body: A PR was merged that solves an issue and awards.csv should be extended.
 | 
			
		||||
							
								
								
									
										54
									
								
								docs/dev/extend-awards.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								docs/dev/extend-awards.md
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,54 @@
 | 
			
		||||
# Automatically extend awards.csv
 | 
			
		||||
 | 
			
		||||
## Overview
 | 
			
		||||
 | 
			
		||||
Whenever a pull request (PR) is merged in the [stacker.news](https://github.com/stackernews/stacker.news) repository, a [GitHub Action](https://docs.github.com/en/actions) is triggered:
 | 
			
		||||
 | 
			
		||||
If the merged PR solves an issue with [award tags](https://github.com/stackernews/stacker.news?tab=readme-ov-file#contributing),
 | 
			
		||||
the amounts due to the PR and issue authors are calculated and corresponding lines are added to the [awards.csv](https://github.com/stackernews/stacker.news/blob/master/awards.csv) file,
 | 
			
		||||
and a PR is opened for this change.
 | 
			
		||||
 | 
			
		||||
## Action
 | 
			
		||||
 | 
			
		||||
The action is defined in [.github/workflows/extend-awards.yml](.github/workflows/extend-awards.yml).
 | 
			
		||||
 | 
			
		||||
Filters on the event type and parameters ensure the action is [triggered only on merged PRs](https://stackoverflow.com/questions/60710209/trigger-github-actions-only-when-pr-is-merged).
 | 
			
		||||
 | 
			
		||||
The primary job consists of several steps:
 | 
			
		||||
  - [checkout](https://github.com/actions/checkout) checks out the repository
 | 
			
		||||
  - [setup-python](https://github.com/actions/setup-python) installs [Python](https://en.wikipedia.org/wiki/Python_(programming_language))
 | 
			
		||||
  - [pip](https://en.wikipedia.org/wiki/Pip_%28package_manager%29) installs the [requests](https://docs.python-requests.org/en/latest/index.html) module
 | 
			
		||||
  - a script (see below) is executed, which appends lines to [awards.csv](awards.csv) if needed
 | 
			
		||||
  - [create-pull-request](https://github.com/peter-evans/create-pull-request) looks for modified files and creates (or updates) a PR
 | 
			
		||||
 | 
			
		||||
## Script
 | 
			
		||||
 | 
			
		||||
The script is [extend-awards.py](extend-awards.py).
 | 
			
		||||
 | 
			
		||||
The script extracts from the [environment](https://en.wikipedia.org/wiki/Environment_variable) an authentication token needed for the [GitHub REST API](https://docs.github.com/en/rest/about-the-rest-api/about-the-rest-api) and the [context](https://docs.github.com/en/actions/writing-workflows/choosing-what-your-workflow-does/accessing-contextual-information-about-workflow-runs) containing the event details including the merged PR (formatted in [JSON](https://en.wikipedia.org/wiki/JSON)).
 | 
			
		||||
 | 
			
		||||
In the merged PR's title and body it searches for the first [GitHub issue URL](https://github.com/stackernews/stacker.news/issues/) or any number with a hash symbol (#) prefix, and takes this as the issue being solved by the PR.
 | 
			
		||||
 | 
			
		||||
Using the GitHub REST API it fetches the issue and analyzes its tags for difficulty and priority.
 | 
			
		||||
 | 
			
		||||
It fetches the issue's timeline and counts the number of reviews completed with status 'changes requested' to calculate the amount reduction.
 | 
			
		||||
 | 
			
		||||
It calculates the amounts due to the PR author and the issue author.
 | 
			
		||||
 | 
			
		||||
It reads the existing awards.csv file to suppress appending redundant lines (same user, PR, and issue) and fill known receive methods (same user).
 | 
			
		||||
 | 
			
		||||
Finally, it appends zero, one, or two lines to the awards.csv file.
 | 
			
		||||
 | 
			
		||||
## Diagnostics
 | 
			
		||||
 | 
			
		||||
In the GitHub web interface under 'Actions' each invokation of the action can be viewed, including environment and [output and errors](https://en.wikipedia.org/wiki/Standard_streams) of the script. First, the specific invokation is selected, then the job 'if_merged', then the step 'Run python extend-awards.py'. The environment is found by expanding the inner 'Run python extended-awards.py' on the first line.
 | 
			
		||||
 | 
			
		||||
The normal output includes details about the issue number found, the amount calculation, or the reason for not appending lines.
 | 
			
		||||
 | 
			
		||||
The error output may include a [Python traceback](https://realpython.com/python-traceback/) which helps to explain the error.
 | 
			
		||||
 | 
			
		||||
The environment contains in GITHUB_CONTEXT the event details, which may be required to understand the error.
 | 
			
		||||
 | 
			
		||||
## Security considerations
 | 
			
		||||
 | 
			
		||||
The create-pull-request step requires [workflow permissions](https://github.com/peter-evans/create-pull-request#workflow-permissions).
 | 
			
		||||
							
								
								
									
										104
									
								
								extend-awards.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								extend-awards.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,104 @@
 | 
			
		||||
import json, os, re, requests
 | 
			
		||||
 | 
			
		||||
difficulties = {'good-first-issue':20000,'easy':100000,'medium':250000,'medium-hard':500000,'hard':1000000}
 | 
			
		||||
priorities = {'low':0.5,'medium':1.5,'high':2,'urgent':3}
 | 
			
		||||
ignored = ['huumn', 'ekzyis']
 | 
			
		||||
fn = 'awards.csv'
 | 
			
		||||
 | 
			
		||||
sess = requests.Session()
 | 
			
		||||
headers = {'Authorization':'Bearer %s' % os.getenv('GITHUB_TOKEN') }
 | 
			
		||||
awards = []
 | 
			
		||||
 | 
			
		||||
def getIssue(n):
 | 
			
		||||
	url = 'https://api.github.com/repos/stackernews/stacker.news/issues/' + n
 | 
			
		||||
	r = sess.get(url, headers=headers)
 | 
			
		||||
	j = json.loads(r.text)
 | 
			
		||||
	return j
 | 
			
		||||
 | 
			
		||||
def findIssueInPR(j):
 | 
			
		||||
	p = re.compile('(#|https://github.com/stackernews/stacker.news/issues/)([0-9]+)')
 | 
			
		||||
	for m in p.finditer(j['title']):
 | 
			
		||||
		return m.group(2)
 | 
			
		||||
	if not 'body' in j or j['body'] is None:
 | 
			
		||||
		return
 | 
			
		||||
	for s in j['body'].split('\n'):
 | 
			
		||||
		for m in p.finditer(s):
 | 
			
		||||
			return m.group(2)
 | 
			
		||||
 | 
			
		||||
def addAward(user, kind, pr, issue, difficulty, priority, count, amount):
 | 
			
		||||
	if amount >= 1000000 and amount % 1000000 == 0:
 | 
			
		||||
		amount = str(int(amount / 1000000)) + 'm'
 | 
			
		||||
	elif amount >= 1000 and amount % 1000 == 0:
 | 
			
		||||
		amount = str(int(amount / 1000)) + 'k'
 | 
			
		||||
	for a in awards:
 | 
			
		||||
		if a[0] == user and a[1] == kind and a[2] == pr:
 | 
			
		||||
			print('found existing entry %s' % a)
 | 
			
		||||
			if a[8] != amount:
 | 
			
		||||
				print('warning: amount %s != %s' % (a[8], amount))
 | 
			
		||||
			return
 | 
			
		||||
	if count < 1:
 | 
			
		||||
		count = ''
 | 
			
		||||
	addr = '???'
 | 
			
		||||
	for a in awards:
 | 
			
		||||
		if a[0] == user and a[9] != '???':
 | 
			
		||||
			addr = a[9]
 | 
			
		||||
	print('adding %s,%s,%s,%s,%s,%s,%s,,%s,%s,???' % (user, kind, pr, issue, difficulty, priority, count, amount, addr))
 | 
			
		||||
	with open(fn, 'a') as f:
 | 
			
		||||
		print('%s,%s,%s,%s,%s,%s,%s,,%s,%s,???' % (user, kind, pr, issue, difficulty, priority, count, amount, addr), file=f)
 | 
			
		||||
 | 
			
		||||
def countReviews(pr):
 | 
			
		||||
	url = 'https://api.github.com/repos/stackernews/stacker.news/issues/%s/timeline' % pr
 | 
			
		||||
	r = sess.get(url, headers=headers)
 | 
			
		||||
	j = json.loads(r.text)
 | 
			
		||||
	count = 0
 | 
			
		||||
	for e in j:
 | 
			
		||||
		if e['event'] == 'reviewed' and e['state'] == 'changes_requested':
 | 
			
		||||
			count += 1
 | 
			
		||||
	return count
 | 
			
		||||
 | 
			
		||||
def checkPR(i):
 | 
			
		||||
	pr = str(i['number'])
 | 
			
		||||
	print('pr %s' % pr)
 | 
			
		||||
	n = findIssueInPR(i)
 | 
			
		||||
	if not n:
 | 
			
		||||
		print('pr %s does not solve an issue' % pr)
 | 
			
		||||
		return
 | 
			
		||||
	print('solves issue %s' % n)
 | 
			
		||||
	j = getIssue(n)
 | 
			
		||||
	difficulty = ''
 | 
			
		||||
	amount = 0
 | 
			
		||||
	priority = ''
 | 
			
		||||
	multiplier = 1
 | 
			
		||||
	for l in j['labels']:
 | 
			
		||||
		for d in difficulties:
 | 
			
		||||
			if l['name'] == 'difficulty:' + d:
 | 
			
		||||
				difficulty = d
 | 
			
		||||
				amount = difficulties[d]
 | 
			
		||||
		for p in priorities:
 | 
			
		||||
			if l['name'] == 'priority:' + p:
 | 
			
		||||
				priority = p
 | 
			
		||||
				multiplier = priorities[p]
 | 
			
		||||
	if amount * multiplier <= 0:
 | 
			
		||||
		print('issue gives no award')
 | 
			
		||||
		return
 | 
			
		||||
	count = countReviews(pr)
 | 
			
		||||
	if count >= 10:
 | 
			
		||||
		print('too many reviews, no award')
 | 
			
		||||
		return
 | 
			
		||||
	if count > 0:
 | 
			
		||||
		print('%d reviews, %d%% reduction' % (count, count * 10))
 | 
			
		||||
	award = amount * multiplier * (10 - count) / 10
 | 
			
		||||
	print('award is %d' % award)
 | 
			
		||||
	if i['user']['login'] not in ignored:
 | 
			
		||||
		addAward(i['user']['login'], 'pr', '#' + pr, '#' + n, difficulty, priority, count, award)
 | 
			
		||||
	if j['user']['login'] not in ignored:
 | 
			
		||||
		count = 0
 | 
			
		||||
		addAward(j['user']['login'], 'issue', '#' + pr, '#' + n, difficulty, priority, count, int(award / 10))
 | 
			
		||||
 | 
			
		||||
with open(fn, 'r') as f:
 | 
			
		||||
	for s in f:
 | 
			
		||||
		s = s.split('\n')[0]
 | 
			
		||||
		awards.append(s.split(','))
 | 
			
		||||
 | 
			
		||||
j = json.loads(os.getenv('GITHUB_CONTEXT'))
 | 
			
		||||
checkPR(j['event']['pull_request'])
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user