Skip to content

Commit 678b02e

Browse files
authored
feat: repository custom properties plugin (#626)
* feat: repository custom properties plugin * docs: custom properties
1 parent b6afb1a commit 678b02e

File tree

6 files changed

+211
-2
lines changed

6 files changed

+211
-2
lines changed

README.md

+6
Original file line numberDiff line numberDiff line change
@@ -498,6 +498,12 @@ branches:
498498
users: []
499499
teams: []
500500

501+
# Custom properties
502+
# See https://docs.github.com/en/rest/repos/custom-properties?apiVersion=2022-11-28
503+
custom_properties:
504+
- name: test
505+
value: test
506+
501507
# See the docs (https://docs.github.com/en/repositories/managing-your-repositorys-settings-and-features/managing-repository-settings/configuring-autolinks-to-reference-external-resources) for a description of autolinks and replacement values.
502508
autolinks:
503509
- key_prefix: 'JIRA-'

app.yml

+1
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ default_events:
3030
# the value (for example, write).
3131
# Valid values are `read`, `write`, and `none`
3232
default_permissions:
33+
repository_custom_properties: write
3334
organization_custom_properties: admin
3435

3536
# Repository creation, deletion, settings, teams, and collaborators.

docs/sample-settings/repo.yml

+7-1
Original file line numberDiff line numberDiff line change
@@ -196,7 +196,13 @@ branches:
196196
apps: []
197197
users: []
198198
teams: []
199-
199+
200+
# Custom properties
201+
# See https://docs.github.com/en/rest/repos/custom-properties?apiVersion=2022-11-28
202+
custom_properties:
203+
- name: test
204+
value: test
205+
200206
validator:
201207
pattern: '[a-zA-Z0-9_-]+_[a-zA-Z0-9_-]+.*'
202208

lib/plugins/custom_properties.js

+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
const Diffable = require('./diffable')
2+
3+
module.exports = class CustomProperties extends Diffable {
4+
constructor (...args) {
5+
super(...args)
6+
7+
if (this.entries) {
8+
// Force all names to lowercase to avoid comparison issues.
9+
this.entries.forEach(prop => {
10+
prop.name = prop.name.toLowerCase()
11+
})
12+
}
13+
}
14+
15+
async find () {
16+
const data = await this.github.request('GET /repos/:org/:repo/properties/values', {
17+
org: this.repo.owner,
18+
repo: this.repo.repo
19+
})
20+
21+
const properties = data.data.map(d => { return { name: d.property_name, value: d.value } })
22+
return properties
23+
}
24+
25+
comparator (existing, attrs) {
26+
return existing.name === attrs.name
27+
}
28+
29+
changed (existing, attrs) {
30+
return attrs.value !== existing.value
31+
}
32+
33+
async update (existing, attrs) {
34+
await this.github.request('PATCH /repos/:org/:repo/properties/values', {
35+
org: this.repo.owner,
36+
repo: this.repo.repo,
37+
properties: [{
38+
property_name: attrs.name,
39+
value: attrs.value
40+
}]
41+
})
42+
}
43+
44+
async add (attrs) {
45+
await this.github.request('PATCH /repos/:org/:repo/properties/values', {
46+
org: this.repo.owner,
47+
repo: this.repo.repo,
48+
properties: [{
49+
property_name: attrs.name,
50+
value: attrs.value
51+
}]
52+
})
53+
}
54+
55+
async remove (existing) {
56+
await this.github.request('PATCH /repos/:org/:repo/properties/values', {
57+
org: this.repo.owner,
58+
repo: this.repo.repo,
59+
properties: [
60+
{
61+
property_name: existing.name,
62+
value: null
63+
}
64+
]
65+
})
66+
}
67+
}

lib/settings.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -836,7 +836,8 @@ Settings.PLUGINS = {
836836
autolinks: require('./plugins/autolinks'),
837837
validator: require('./plugins/validator'),
838838
rulesets: require('./plugins/rulesets'),
839-
environments: require('./plugins/environments')
839+
environments: require('./plugins/environments'),
840+
custom_properties: require('./plugins/custom_properties.js')
840841
}
841842

842843
module.exports = Settings
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
const CustomProperties = require('../../../../lib/plugins/custom_properties')
2+
3+
describe('CustomProperties', () => {
4+
let github
5+
const repo = { owner: 'owner', repo: 'repo' }
6+
let log
7+
8+
function configure (config) {
9+
const nop = false;
10+
const errors = []
11+
return new CustomProperties(nop, github, { owner: 'bkeepers', repo: 'test' }, config, log, errors)
12+
}
13+
14+
beforeEach(() => {
15+
github = {
16+
request: jest.fn()
17+
// .mockResolvedValue({
18+
// data: [
19+
// { property_name: 'test', value: 'test' }
20+
// ]
21+
// })
22+
}
23+
log = { debug: jest.fn(), error: console.error }
24+
})
25+
26+
describe('sync', () => {
27+
it('syncs custom properties', async () => {
28+
const plugin = configure([
29+
{ name: 'test', value: 'test' }
30+
])
31+
32+
github.request.mockResolvedValue({
33+
data: [
34+
{ property_name: 'test', value: 'test' }
35+
]
36+
})
37+
38+
return plugin.sync().then(() => {
39+
expect(github.request).toHaveBeenCalledWith('GET /repos/:org/:repo/properties/values', {
40+
org: 'bkeepers',
41+
repo: 'test'
42+
})
43+
})
44+
})
45+
})
46+
describe('sync', () => {
47+
it('add custom properties', async () => {
48+
const plugin = configure([
49+
{ name: 'test', value: 'test' }
50+
])
51+
52+
github.request.mockResolvedValue({
53+
data: []
54+
})
55+
56+
return plugin.sync().then(() => {
57+
expect(github.request).toHaveBeenNthCalledWith(1, 'GET /repos/:org/:repo/properties/values', {
58+
org: 'bkeepers',
59+
repo: 'test'
60+
})
61+
expect(github.request).toHaveBeenNthCalledWith(2, 'PATCH /repos/:org/:repo/properties/values', {
62+
org: 'bkeepers',
63+
repo: 'test',
64+
properties: [
65+
{
66+
property_name: 'test',
67+
value: 'test'
68+
}
69+
]
70+
})
71+
})
72+
})
73+
})
74+
describe('sync', () => {
75+
it('remove custom properties', async () => {
76+
const plugin = configure([])
77+
78+
github.request.mockResolvedValue({
79+
data: [{ property_name: 'test', value: 'test' }]
80+
})
81+
82+
return plugin.sync().then(() => {
83+
expect(github.request).toHaveBeenNthCalledWith(1, 'GET /repos/:org/:repo/properties/values', {
84+
org: 'bkeepers',
85+
repo: 'test'
86+
})
87+
expect(github.request).toHaveBeenNthCalledWith(2, 'PATCH /repos/:org/:repo/properties/values', {
88+
org: 'bkeepers',
89+
repo: 'test',
90+
properties: [
91+
{
92+
property_name: 'test',
93+
value: null
94+
}
95+
]
96+
})
97+
})
98+
})
99+
})
100+
describe('sync', () => {
101+
it('update custom properties', async () => {
102+
const plugin = configure([
103+
{ name: 'test', value: 'foobar' }
104+
])
105+
106+
github.request.mockResolvedValue({
107+
data: [{ property_name: 'test', value: 'test' }]
108+
})
109+
110+
return plugin.sync().then(() => {
111+
expect(github.request).toHaveBeenNthCalledWith(1, 'GET /repos/:org/:repo/properties/values', {
112+
org: 'bkeepers',
113+
repo: 'test'
114+
})
115+
expect(github.request).toHaveBeenNthCalledWith(2, 'PATCH /repos/:org/:repo/properties/values', {
116+
org: 'bkeepers',
117+
repo: 'test',
118+
properties: [
119+
{
120+
property_name: 'test',
121+
value: 'foobar'
122+
}
123+
]
124+
})
125+
})
126+
})
127+
})
128+
})

0 commit comments

Comments
 (0)