Browse Source

Add JIRA session and requests for OAuth. Redirect not handled yet.

pull/17/head
Kegan Dougal 9 years ago
parent
commit
4fdc0c3912
  1. 17
      README.md
  2. 99
      src/github.com/matrix-org/go-neb/realms/jira/jira.go

17
README.md

@ -199,7 +199,22 @@ JIRA installation. Once that is complete, users can OAuth on the target JIRA ins
### Make a request for JIRA Auth ### Make a request for JIRA Auth
TODO
```
curl -X POST localhost:4050/admin/requestAuthSession --data-binary '{
"RealmID": "jirarealm",
"UserID": "@your_user_id:localhost",
"Config": {
}
}'
```
Returns:
```json
{
"URL":"https://jira.somewhere.com/plugins/servlet/oauth/authorize?oauth_token=7yeuierbgweguiegrTbOT"
}
```
Follow this link and grant access for NEB to act on your behalf.
### Create a JIRA bot ### Create a JIRA bot

99
src/github.com/matrix-org/go-neb/realms/jira/jira.go

@ -9,6 +9,7 @@ import (
log "github.com/Sirupsen/logrus" log "github.com/Sirupsen/logrus"
"github.com/andygrunwald/go-jira" "github.com/andygrunwald/go-jira"
"github.com/dghubble/oauth1" "github.com/dghubble/oauth1"
"github.com/matrix-org/go-neb/database"
"github.com/matrix-org/go-neb/realms/jira/urls" "github.com/matrix-org/go-neb/realms/jira/urls"
"github.com/matrix-org/go-neb/types" "github.com/matrix-org/go-neb/types"
"golang.org/x/net/context" "golang.org/x/net/context"
@ -28,6 +29,25 @@ type jiraRealm struct {
PrivateKeyPEM string PrivateKeyPEM string
} }
type JIRASession struct {
id string // request token
userID string
realmID string
Secret string // request secret
}
func (s *JIRASession) UserID() string {
return s.userID
}
func (s *JIRASession) RealmID() string {
return s.realmID
}
func (s *JIRASession) ID() string {
return s.id
}
func (r *jiraRealm) ID() string { func (r *jiraRealm) ID() string {
return r.id return r.id
} }
@ -78,7 +98,45 @@ func (r *jiraRealm) Register() error {
} }
func (r *jiraRealm) RequestAuthSession(userID string, req json.RawMessage) interface{} { func (r *jiraRealm) RequestAuthSession(userID string, req json.RawMessage) interface{} {
logger := log.WithField("jira_url", r.JIRAEndpoint)
// Parse the private key as we may not have called Register()
err := r.parsePrivateKey()
if err != nil {
logger.WithError(err).Print("Failed to parse private key")
return nil
}
ju, err := urls.ParseJIRAURL(r.JIRAEndpoint)
if err != nil {
log.WithError(err).Print("Failed to parse JIRA endpoint")
return nil
}
authConfig := r.oauth1Config(ju)
reqToken, reqSec, err := authConfig.RequestToken()
if err != nil {
logger.WithError(err).Print("Failed to request auth token")
return nil
}
logger.WithField("req_token", reqToken).Print("Received request token")
authURL, err := authConfig.AuthorizationURL(reqToken)
if err != nil {
logger.WithError(err).Print("Failed to create authorization URL")
return nil
}
_, err = database.GetServiceDB().StoreAuthSession(&JIRASession{
id: reqToken,
userID: userID,
realmID: r.id,
Secret: reqSec,
})
if err != nil {
log.WithError(err).Print("Failed to store new auth session")
return nil return nil
}
return &struct {
URL string
}{authURL.String()}
} }
func (r *jiraRealm) OnReceiveRedirect(w http.ResponseWriter, req *http.Request) { func (r *jiraRealm) OnReceiveRedirect(w http.ResponseWriter, req *http.Request) {
@ -99,24 +157,7 @@ func (r *jiraRealm) jiraClient(u urls.JIRAURL, userID string, allowUnauth bool)
// make an authenticated client // make an authenticated client
var cli *jira.Client var cli *jira.Client
auth := &oauth1.Config{
ConsumerKey: r.ConsumerKey,
ConsumerSecret: r.ConsumerSecret,
CallbackURL: u.Base + "realms/redirect/" + r.id,
// TODO: In JIRA Cloud, the Authorization URL is only the Instance BASE_URL:
// https://BASE_URL.atlassian.net.
// It also does not require the + "/plugins/servlet/oauth/authorize"
// We should probably check the provided JIRA base URL to see if it is a cloud one
// then adjust accordingly.
Endpoint: oauth1.Endpoint{
RequestTokenURL: u.Base + "plugins/servlet/oauth/request-token",
AuthorizeURL: u.Base + "plugins/servlet/oauth/authorize",
AccessTokenURL: u.Base + "plugins/servlet/oauth/access-token",
},
Signer: &oauth1.RSASigner{
PrivateKey: r.privateKey,
},
}
auth := r.oauth1Config(u)
httpClient := auth.Client(context.TODO(), oauth1.NewToken("access_tokenTODO", "access_secretTODO")) httpClient := auth.Client(context.TODO(), oauth1.NewToken("access_tokenTODO", "access_secretTODO"))
cli, err := jira.NewClient(httpClient, u.Base) cli, err := jira.NewClient(httpClient, u.Base)
@ -144,6 +185,28 @@ func (r *jiraRealm) parsePrivateKey() error {
return nil return nil
} }
func (r *jiraRealm) oauth1Config(u urls.JIRAURL) *oauth1.Config {
return &oauth1.Config{
ConsumerKey: r.ConsumerKey,
ConsumerSecret: r.ConsumerSecret,
// TODO: path from goneb.go - we should factor it out like we did with Services
CallbackURL: u.Base + "realms/redirect/" + r.id,
// TODO: In JIRA Cloud, the Authorization URL is only the Instance BASE_URL:
// https://BASE_URL.atlassian.net.
// It also does not require the + "/plugins/servlet/oauth/authorize"
// We should probably check the provided JIRA base URL to see if it is a cloud one
// then adjust accordingly.
Endpoint: oauth1.Endpoint{
RequestTokenURL: u.Base + "plugins/servlet/oauth/request-token",
AuthorizeURL: u.Base + "plugins/servlet/oauth/authorize",
AccessTokenURL: u.Base + "plugins/servlet/oauth/access-token",
},
Signer: &oauth1.RSASigner{
PrivateKey: r.privateKey,
},
}
}
func loadPrivateKey(privKeyPEM string) (*rsa.PrivateKey, error) { func loadPrivateKey(privKeyPEM string) (*rsa.PrivateKey, error) {
// Decode PEM to grab the private key type // Decode PEM to grab the private key type
block, _ := pem.Decode([]byte(privKeyPEM)) block, _ := pem.Decode([]byte(privKeyPEM))

Loading…
Cancel
Save