47degrees / github4s

A GitHub API wrapper written in Scala
http://47degrees.github.io/github4s
Apache License 2.0
229 stars 75 forks source link

GithubApp Auth #554

Open loonydev opened 4 years ago

loonydev commented 4 years ago

Hi, As I understand, for now this library don't support GithubApps authorization?

If so, I want to discuss the best way for implementation.

In my code I used trait GithubAuth with method def getAuthHeader():String. GithubClient taken instance of GithubAuth and call this function for token. In case of PAT it's return token from constructor, in case of private key it's looks like:

class GithubKeyAuth(privateKeyContent: String, appId: String) extends GithubAuth {

  implicit val clock: Clock = Clock.systemUTC
  // build PrivateKey instance from string
  private val privateKey = GithubKeyAuth.getPrivateKey(privateKeyContent)

 // generate valid JWT token using PrivateKey
  override def getAuthHeader(): String = {
    "Bearer " + Jwt.encode(JwtClaim({
      s"""{"iss":$appId}"""
    }).issuedNow.expiresIn(10), privateKey, JwtAlgorithm.RS256)
  }
}

WDYT?

BenFradet commented 4 years ago

You're right we don't support github apps authorization. However, I'm not sure about the best way to go about it as we don't want to handle jwt tokens directly in the library for example.

loonydev commented 4 years ago

Agreed, maybe we can make GithubClient to accept instance of GithubAuth, so the user can use by default GithubPatAuth, in other cases they need to define own class.

But if it's part of Github Auth flow user will do same code over and over.

loonydev commented 4 years ago

So, do you have any suggestions how to implement it? I can do that, by I need to be sure that I don't waste a time.

BenFradet commented 4 years ago

maybe we can make GithubClient to accept instance of GithubAuth, so the user can use by default GithubPatAuth, in other cases they need to define own class.

I'm :+1: on that

loonydev commented 4 years ago

But as I mention before, users will do same code over and over and not in the best way. It's like give them domain and say, use own http sender.

BenFradet commented 4 years ago

We can add that code as documentation, it doesn't seem to me like the usage will be so huge that it would need to be part of the library.

loonydev commented 4 years ago

Okay, next week I'm going to prepare this changes and will discuss it with real example.

kusaeva commented 4 years ago

@loonydev Hi! Are you still planning to make a PR with this changes?

loonydev commented 4 years ago

Morning @kusaeva, not yet, it's in plan, but not in a near future. If you have a time to make it, it would be great.

nmcb commented 1 year ago

Hello - I'm interested about the status of this issue

We have a standalone service in our infrastructure that requires broad access to GH's REST api and the repositories under our organisation (DHL-Parcel) - preferably using the github4s library and authorise not via a personal access token - but either via JWT tokens or - as requested in this issue - via GH Apps authorisation.

We did try the new AccessToken feature to get this working but weren't able to. The main issue seems to be the Authorisation: token header being hardcoded in the library, but we might not understand the way the new features was intended to be used. In that case an example would help us.

If authorisation via JWT or GH App tokens is currently not possible we would able to spend some time on getting this to work with github4s provided that we get some help and pointers on how the preferred integration should look like in the API.

Thanks in advance :)

fedefernandez commented 1 year ago

Hi @nmcb, thanks for your interest.

This is how it's currently built.

We have an algebra that allows generating tokens:

https://github.com/47degrees/github4s/blob/9c18225cc61f756aace9ddcb5d134720c882edb0/github4s/shared/src/main/scala/github4s/algebras/AccessToken.scala#L31-L34

This algebra has a single implementation called StaticAccessToken. You pass a token, and it returns as a pure value.

https://github.com/47degrees/github4s/blob/9c18225cc61f756aace9ddcb5d134720c882edb0/github4s/shared/src/main/scala/github4s/interpreters/StaticAccessToken.scala#L25-L29

When you create a new Github4s client, it's creates an instance of that algebra to use it internally:

https://github.com/47degrees/github4s/blob/9c18225cc61f756aace9ddcb5d134720c882edb0/github4s/shared/src/main/scala/github4s/Github.scala#L53

This is then used to create the internal GithubAPIv3 algebra:

https://github.com/47degrees/github4s/blob/9c18225cc61f756aace9ddcb5d134720c882edb0/github4s/shared/src/main/scala/github4s/Github.scala#L31

That, in a similar way, pass it to the internal HTTPClient algebra:

https://github.com/47degrees/github4s/blob/9c18225cc61f756aace9ddcb5d134720c882edb0/github4s/shared/src/main/scala/github4s/modules/GithubAPIs.scala#L32

The HTTPClient uses the withAccessToken to get the token and pass it to the RequestBuilder

https://github.com/47degrees/github4s/blob/9c18225cc61f756aace9ddcb5d134720c882edb0/github4s/shared/src/main/scala/github4s/http/HttpClient.scala#L65-L69

The RequestBuilder stores the token in a authHeader map, where the key is Authorization, and the value is s"token $token".

https://github.com/47degrees/github4s/blob/9c18225cc61f756aace9ddcb5d134720c882edb0/github4s/shared/src/main/scala/github4s/http/RequestBuilder.scala#L44-L48

This is mapped to a Header.Raw

https://github.com/47degrees/github4s/blob/9c18225cc61f756aace9ddcb5d134720c882edb0/github4s/shared/src/main/scala/github4s/http/Http4sSyntax.scala#L50

Potential solution

@loonydev's approach looks good, so I've created a draft with a potential solution:

In that way, you can implement your algebra for generating auth headers, having complete control. Let me know your thoughts. The docs need to be updated accordingly.

nmcb commented 1 year ago

thanks @fedefernandez! we'll give it a try.

cc @thijsnissen