2017-02-11 22:27:47 +01:00
import * as EventEmitter from 'events' ;
2017-01-31 16:12:49 +01:00
import * as express from 'express' ;
2017-02-11 22:39:37 +01:00
const crypto = require ( 'crypto' ) ;
2017-01-31 16:43:06 +01:00
import User from '../models/user' ;
2018-03-28 18:20:40 +02:00
import config from '../../../conf' ;
import queue from '../../../queue' ;
2017-01-31 16:12:49 +01:00
2017-01-31 16:43:06 +01:00
module .exports = async ( app : express.Application ) = > {
2017-01-31 16:12:49 +01:00
if ( config . github_bot == null ) return ;
2017-01-31 16:43:06 +01:00
const bot = await User . findOne ( {
2018-03-29 07:48:47 +02:00
usernameLower : config.github_bot.username.toLowerCase ( )
2017-01-31 16:43:06 +01:00
} ) ;
if ( bot == null ) {
console . warn ( ` GitHub hook bot specified, but not found: @ ${ config . github_bot . username } ` ) ;
return ;
}
const post = text = > require ( '../endpoints/posts/create' ) ( { text } , bot ) ;
2017-02-11 22:27:47 +01:00
const handler = new EventEmitter ( ) ;
2017-01-31 16:12:49 +01:00
2017-02-11 22:27:47 +01:00
app . post ( '/hooks/github' , ( req , res , next ) = > {
2017-08-10 15:06:47 +02:00
// req.headers['x-hub-signature'] および
// req.headers['x-github-event'] は常に string ですが、型定義の都合上
// string | string[] になっているので string を明示しています
2018-03-28 18:20:40 +02:00
// if ((new Buffer(req.headers['x-hub-signature'] as string)).equals(new Buffer(`sha1=${crypto.createHmac('sha1', config.github_bot.hook_secret).update(JSON.stringify(req.body)).digest('hex')}`))) {
2017-08-10 15:06:47 +02:00
handler . emit ( req . headers [ 'x-github-event' ] as string , req . body ) ;
2017-02-16 12:08:28 +01:00
res . sendStatus ( 200 ) ;
2018-03-28 18:20:40 +02:00
// } else {
// res.sendStatus(400);
// }
2017-02-11 22:27:47 +01:00
} ) ;
2017-01-31 16:12:49 +01:00
2017-03-17 17:42:43 +01:00
handler . on ( 'status' , event = > {
const state = event . state ;
switch ( state ) {
2017-03-31 13:24:14 +02:00
case 'error' :
2017-03-17 17:42:43 +01:00
case 'failure' :
2017-03-17 17:51:08 +01:00
const commit = event . commit ;
2017-03-17 21:42:07 +01:00
const parent = commit . parents [ 0 ] ;
2018-03-28 18:20:40 +02:00
queue . create ( 'gitHubFailureReport' , {
userId : bot._id ,
parentUrl : parent.url ,
htmlUrl : commit.html_url ,
message : commit.commit.message ,
} ) . save ( ) ;
2017-03-17 17:42:43 +01:00
break ;
}
} ) ;
2017-01-31 20:25:02 +01:00
handler . on ( 'push' , event = > {
2017-02-11 22:44:16 +01:00
const ref = event . ref ;
2017-03-14 01:37:07 +01:00
switch ( ref ) {
case 'refs/heads/master' :
const pusher = event . pusher ;
const compare = event . compare ;
2017-03-16 16:21:32 +01:00
const commits = event . commits ;
post ( [
2017-03-17 17:22:28 +01:00
` Pushed by ** ${ pusher . name } ** with ?[ ${ commits . length } commit ${ commits . length > 1 ? 's' : '' } ]( ${ compare } ): ` ,
2017-03-17 18:47:55 +01:00
commits . reverse ( ) . map ( commit = > ` ・[?[ ${ commit . id . substr ( 0 , 7 ) } ]( ${ commit . url } )] ${ commit . message . split ( '\n' ) [ 0 ] } ` ) . join ( '\n' ) ,
2017-03-16 16:21:32 +01:00
] . join ( '\n' ) ) ;
2017-03-14 01:37:07 +01:00
break ;
case 'refs/heads/release' :
const commit = event . commits [ 0 ] ;
2017-03-16 16:21:32 +01:00
post ( ` RELEASED: ${ commit . message } ` ) ;
2017-03-14 01:37:07 +01:00
break ;
}
2017-01-31 16:12:49 +01:00
} ) ;
2017-01-31 16:43:06 +01:00
handler . on ( 'issues' , event = > {
2017-02-11 22:44:16 +01:00
const issue = event . issue ;
const action = event . action ;
2017-01-31 16:43:06 +01:00
let title : string ;
2017-01-31 20:07:30 +01:00
switch ( action ) {
2017-03-26 14:34:04 +02:00
case 'opened' : title = 'Issue opened' ; break ;
case 'closed' : title = 'Issue closed' ; break ;
case 'reopened' : title = 'Issue reopened' ; break ;
2017-01-31 17:04:32 +01:00
default : return ;
2017-01-31 16:43:06 +01:00
}
2017-03-14 01:37:07 +01:00
post ( ` ${ title } : < ${ issue . number } >「 ${ issue . title } 」 \ n ${ issue . html_url } ` ) ;
2017-01-31 16:43:06 +01:00
} ) ;
2017-01-31 20:25:02 +01:00
handler . on ( 'issue_comment' , event = > {
2017-02-11 22:44:16 +01:00
const issue = event . issue ;
const comment = event . comment ;
const action = event . action ;
2017-01-31 20:25:02 +01:00
let text : string ;
switch ( action ) {
2017-03-14 01:37:07 +01:00
case 'created' : text = ` Commented to「 ${ issue . title } 」: ${ comment . user . login } 「 ${ comment . body } 」 \ n ${ comment . html_url } ` ; break ;
2017-01-31 20:25:02 +01:00
default : return ;
}
post ( text ) ;
} ) ;
2017-02-17 02:25:23 +01:00
handler . on ( 'watch' , event = > {
2017-02-11 22:44:16 +01:00
const sender = event . sender ;
2017-08-28 19:56:07 +02:00
post ( ` ⭐️ Starred by ** ${ sender . login } ** ⭐️ ` ) ;
2017-02-07 15:52:58 +01:00
} ) ;
2017-01-31 20:25:02 +01:00
handler . on ( 'fork' , event = > {
2017-02-11 22:44:16 +01:00
const repo = event . forkee ;
2017-08-28 19:56:07 +02:00
post ( ` 🍴 Forked: \ n ${ repo . html_url } 🍴 ` ) ;
2017-01-31 20:25:02 +01:00
} ) ;
handler . on ( 'pull_request' , event = > {
2017-02-11 22:44:16 +01:00
const pr = event . pull_request ;
const action = event . action ;
2017-01-31 20:25:02 +01:00
let text : string ;
switch ( action ) {
case 'opened' : text = ` New Pull Request:「 ${ pr . title } 」 \ n ${ pr . html_url } ` ; break ;
case 'reopened' : text = ` Pull Request Reopened:「 ${ pr . title } 」 \ n ${ pr . html_url } ` ; break ;
case 'closed' :
text = pr . merged
? ` Pull Request Merged!:「 ${ pr . title } 」 \ n ${ pr . html_url } `
: ` Pull Request Closed:「 ${ pr . title } 」 \ n ${ pr . html_url } ` ;
break ;
default : return ;
}
post ( text ) ;
} ) ;
2017-01-31 16:12:49 +01:00
} ;