Skip to content

Commit

Permalink
Add alternative GH auth: GH app
Browse files Browse the repository at this point in the history
  • Loading branch information
ledoyen committed Nov 23, 2024
1 parent 3ef3ee2 commit b6d1855
Show file tree
Hide file tree
Showing 9 changed files with 221 additions and 78 deletions.
1 change: 0 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -12,4 +12,3 @@ buildNumber.properties
# IntelliJ
.idea/
*.iml

25 changes: 25 additions & 0 deletions pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,8 @@
<amqp-client.version>5.23.0</amqp-client.version>
<juniversalchardet.version>2.5.0</juniversalchardet.version>
<testcontainers.version>1.20.4</testcontainers.version>
<jjwt.version>0.12.3</jjwt.version>
<bouncycastle.version>1.79</bouncycastle.version>

<junit-jupiter.version>5.11.3</junit-jupiter.version>
<assertj-core.version>3.26.3</assertj-core.version>
Expand Down Expand Up @@ -168,6 +170,29 @@
<artifactId>testcontainers</artifactId>
<version>${testcontainers.version}</version>
</dependency>
<dependency>
<groupId>io.jsonwebtoken</groupId>
<artifactId>jjwt-api</artifactId>
<version>${jjwt.version}</version>
</dependency>
<dependency>
<groupId>io.jsonwebtoken</groupId>
<artifactId>jjwt-impl</artifactId>
<version>${jjwt.version}</version>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>io.jsonwebtoken</groupId>
<artifactId>jjwt-jackson</artifactId>
<version>${jjwt.version}</version>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>org.bouncycastle</groupId>
<artifactId>bcprov-jdk18on</artifactId>
<version>${bouncycastle.version}</version>
</dependency>


<!-- Test dependencies -->
<dependency>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,7 @@ class GradingJob(
val failedSlugs = mutableListOf<String>()
val jobDurations = mutableListOf<Long>()
for ((jobIndex, userSlug) in userSlugs.withIndex()) {
System.setProperty("github_user", userSlug)

Check warning on line 71 in src/main/kotlin/com/github/lernejo/korekto/toolkit/GradingJob.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/GradingJob.kt#L71

Added line #L71 was not covered by tests
val gradingConfiguration = GradingConfiguration(
repoUrlBuilder(userSlug),
"",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,7 @@ class GradingJobLauncher : Callable<Int> {
0
}
slug.isPresent -> {
System.setProperty("github_user", slug.get())

Check warning on line 71 in src/main/kotlin/com/github/lernejo/korekto/toolkit/launcher/GradingJobLauncher.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/launcher/GradingJobLauncher.kt#L71

Added line #L71 was not covered by tests
val repoUrl = grader.slugToRepoUrl(slug.get())
val configuration = GradingConfiguration(repoUrl, "", "")
val branch = System.getProperty("git.branch")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ class ExerciseCloner(private val workspace: Path) {
if (potentialRepo.isPresent) {
if (!forcePull) {
try {
forcePull(potentialRepo.get(), uri)
forcePull(potentialRepo.get())

Check warning on line 41 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/git/ExerciseCloner.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/git/ExerciseCloner.kt#L41

Added line #L41 was not covered by tests
} catch (e: RuntimeException) {
throw RuntimeException("Could not pull -f repository: $path ($uri), ${e.message}", e)
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,14 +1,14 @@
package com.github.lernejo.korekto.toolkit.thirdparty.git

import com.github.lernejo.korekto.toolkit.WarningException
import com.github.lernejo.korekto.toolkit.thirdparty.github.GitHubAuthenticationHolder
import org.eclipse.jgit.api.Git
import org.eclipse.jgit.api.ResetCommand
import org.eclipse.jgit.api.errors.GitAPIException
import org.eclipse.jgit.api.errors.InvalidRemoteException
import org.eclipse.jgit.api.errors.JGitInternalException
import org.eclipse.jgit.api.errors.NoHeadException
import org.eclipse.jgit.storage.file.FileRepositoryBuilder
import org.eclipse.jgit.transport.UsernamePasswordCredentialsProvider
import org.slf4j.LoggerFactory
import java.io.IOException
import java.io.UncheckedIOException
Expand All @@ -20,7 +20,6 @@ object GitRepository {
private val LOGGER = LoggerFactory.getLogger(GitRepository::class.java)

private val URI_WITH_CRED_PATTERN = Pattern.compile("(?<protocol>https?://)(?<cred>[^@]+)@(?<hostAndMore>.+)")
private val URI_WITHOUT_CRED_PATTERN = Pattern.compile("(?<protocol>https?://)(?<hostAndMore>.+)")

data class Creds(val uriWithoutCred: String, val username: String?, val password: String?)

Expand Down Expand Up @@ -51,42 +50,22 @@ object GitRepository {
}
}

private fun insertToken(uri: String, token: String): String {
val uriCredMatcher = URI_WITH_CRED_PATTERN.matcher(uri)
return if (uriCredMatcher.matches()) {
uri
} else {
val uriMatcher = URI_WITHOUT_CRED_PATTERN.matcher(uri)
if (uriMatcher.matches()) {
uriMatcher.group("protocol") + "x-access-token:" + token + "@" + uriMatcher.group("hostAndMore")
} else {
uri
}
}
}

@JvmStatic
fun clone(uri: String, path: Path): Git {
val creds = extractCredParts(uri)

return try {
val cloneCommand = Git.cloneRepository()
.setURI(uri)
.setDirectory(path.toFile())
val token = System.getProperty("github_token")
if (creds.username != null) {
cloneCommand.setCredentialsProvider(UsernamePasswordCredentialsProvider(creds.username, creds.password))
} else if (token != null) {
cloneCommand
.setURI(insertToken(uri, token))
.setCredentialsProvider(UsernamePasswordCredentialsProvider(token, ""))
}
cloneCommand
.setURI(uri)
GitHubAuthenticationHolder.auth.configure(cloneCommand)

val git = cloneCommand
.setCloneAllBranches(true)
.call()
LOGGER.debug("Cloning in: " + git.repository.directory)
git
} catch(e: InvalidRemoteException) {
} catch (e: InvalidRemoteException) {

Check warning on line 68 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/git/GitRepository.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/git/GitRepository.kt#L68

Added line #L68 was not covered by tests
throw WarningException("Unable to clone in ${path.toAbsolutePath()}: Missing or inaccessible repository", e)
} catch (e: GitAPIException) {
throw buildWarningException(path, e)
Expand Down Expand Up @@ -122,38 +101,12 @@ object GitRepository {
}
}

internal fun extractCreds(uri: String): Creds {
val token = System.getProperty("github_token")
return if (token != null) {
Creds(uri, token, "")
} else {
extractCredParts(uri)
}
}

@JvmStatic
fun forcePull(git: Git, uri: String) {
val creds = extractCreds(uri)

try {
try {
forcePull(git, creds, "origin/master")
} catch (e: JGitInternalException) {
forcePull(git, creds, "origin/main")
}
} catch (e: GitAPIException) {
throw RuntimeException(e)
}
}

fun forcePull(git: Git, creds: Creds, ref: String) {
fun forcePull(git: Git) {
val fetchCommand = git.fetch()
.setForceUpdate(true)
if (creds.username != null) {
fetchCommand.setCredentialsProvider(UsernamePasswordCredentialsProvider(creds.username, creds.password))
}
fetchCommand
.call()
GitHubAuthenticationHolder.auth.configure(fetchCommand)
fetchCommand.call()

Check warning on line 109 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/git/GitRepository.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/git/GitRepository.kt#L108-L109

Added lines #L108 - L109 were not covered by tests
git.reset().setMode(ResetCommand.ResetType.HARD).call()
git.clean().setCleanDirectories(true).setForce(true).call()
git.pull().call()
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,167 @@
package com.github.lernejo.korekto.toolkit.thirdparty.github

import io.jsonwebtoken.JwtBuilder
import io.jsonwebtoken.Jwts
import io.jsonwebtoken.security.SecureDigestAlgorithm
import org.bouncycastle.jce.provider.BouncyCastleProvider
import org.eclipse.jgit.api.GitCommand
import org.eclipse.jgit.api.TransportCommand
import org.eclipse.jgit.transport.UsernamePasswordCredentialsProvider
import org.kohsuke.github.GitHubBuilder
import org.testcontainers.shaded.org.bouncycastle.openssl.PEMKeyPair
import org.testcontainers.shaded.org.bouncycastle.openssl.PEMParser
import org.testcontainers.shaded.org.bouncycastle.openssl.jcajce.JcaPEMKeyConverter
import java.io.File
import java.io.FileReader
import java.net.HttpURLConnection
import java.security.PrivateKey
import java.security.PublicKey
import java.security.Security
import java.time.Instant
import java.util.*

object GitHubAuthenticationHolder {
val auth: GitHubAuthentication by lazy {
val token = System.getProperty("github_token")
val appId = System.getProperty("github_app_id")
val appPk = System.getProperty("github_app_pk")
if (token != null) {
TokenGitHubAuthentication(token)
} else if (appId != null && appPk != null) {
AppGitHubAuthentication(appId, appPk)

Check warning on line 31 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L31

Added line #L31 was not covered by tests
} else {
NoopTokenGitHubAuthentication()

Check warning on line 33 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L33

Added line #L33 was not covered by tests
}
}
}

interface GitHubAuthentication {
val type: String
fun <R, C : GitCommand<R>> configure(command: TransportCommand<in C, R>)
fun configure(conn: HttpURLConnection)
fun configure(builder: GitHubBuilder)
}

class NoopTokenGitHubAuthentication : GitHubAuthentication {
override val type = "noop"

Check warning on line 46 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L45-L46

Added lines #L45 - L46 were not covered by tests

override fun <R, C : GitCommand<R>> configure(command: TransportCommand<in C, R>) {
}

Check warning on line 49 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L49

Added line #L49 was not covered by tests

override fun configure(conn: HttpURLConnection) {
}

Check warning on line 52 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L52

Added line #L52 was not covered by tests

override fun configure(builder: GitHubBuilder) {
}

Check warning on line 55 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L55

Added line #L55 was not covered by tests
}

data class TokenGitHubAuthentication(val token: String) : GitHubAuthentication {
override val type = "token"

override fun <R, C : GitCommand<R>> configure(command: TransportCommand<in C, R>) {
command
.setCredentialsProvider(UsernamePasswordCredentialsProvider(token, ""))
}

override fun configure(conn: HttpURLConnection) {
conn.setRequestProperty("Authorization", "token $token")
}

override fun configure(builder: GitHubBuilder) {
builder.withOAuthToken(token)
}
}

class AppGitHubAuthentication(private val appId: String, appPk: String) : GitHubAuthentication {
override val type = "app-$appId (installation: " + getToken().installationId + ")"

Check warning on line 76 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L75-L76

Added lines #L75 - L76 were not covered by tests

companion object {
init {
Security.removeProvider("BC") //remove old/legacy Android-provided BC provider
Security.addProvider(BouncyCastleProvider()) // add 'real'/correct BC provider
}

Check warning on line 82 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L79-L82

Added lines #L79 - L82 were not covered by tests
}

private val privateKey = readPrivateKey(appPk)

Check warning on line 85 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L85

Added line #L85 was not covered by tests
private var jwt: Jwt? = null
private val tokensByUser: MutableMap<String, InstallationToken> = mutableMapOf()

Check warning on line 87 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L87

Added line #L87 was not covered by tests

override fun <R, C : GitCommand<R>> configure(command: TransportCommand<in C, R>) {
command.setCredentialsProvider(UsernamePasswordCredentialsProvider("x-access-token", getToken().token))
}

Check warning on line 91 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L90-L91

Added lines #L90 - L91 were not covered by tests

override fun configure(conn: HttpURLConnection) {
val token = getToken().token
conn.setRequestProperty("Authorization", "Bearer $token")
}

Check warning on line 96 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L94-L96

Added lines #L94 - L96 were not covered by tests

override fun configure(builder: GitHubBuilder) {
builder.withAppInstallationToken(getToken().token)
}

Check warning on line 100 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L99-L100

Added lines #L99 - L100 were not covered by tests

private fun getToken(): InstallationToken {
val user = System.getProperty("github_user") ?: throw IllegalStateException("Missing github_user env prop")
val token = tokensByUser[user]

Check warning on line 104 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L104

Added line #L104 was not covered by tests
if (token == null || token.isExpired()) {
tokensByUser[user] = refreshToken(user)

Check warning on line 106 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L106

Added line #L106 was not covered by tests
}
return tokensByUser[user]!!

Check warning on line 108 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L108

Added line #L108 was not covered by tests
}

private fun refreshToken(user: String): InstallationToken {
val jwt = getJwt()
val gitHubApp = GitHubBuilder().withJwtToken(jwt).build()
val installation = gitHubApp.app.getInstallationByUser(user)

Check warning on line 114 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L112-L114

Added lines #L112 - L114 were not covered by tests

val tokenResponse = installation.createToken().create()

Check warning on line 116 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L116

Added line #L116 was not covered by tests

return InstallationToken(tokenResponse.token, installation.id, tokenResponse.expiresAt.toInstant())

Check warning on line 118 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L118

Added line #L118 was not covered by tests
}

private fun getJwt(): String {
if (jwt == null || jwt!!.isExpired()) {
jwt = createJWT(appId, 590000, privateKey)

Check warning on line 123 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L123

Added line #L123 was not covered by tests
}
return jwt!!.token

Check warning on line 125 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L125

Added line #L125 was not covered by tests
}
}

class Jwt(val token: String, private val start: Long, private val duration: Long) {

Check warning on line 129 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L129

Added line #L129 was not covered by tests
fun isExpired() = (start + duration - 10_000) > System.currentTimeMillis()
}

class InstallationToken(val token: String, val installationId: Long, private val expiresAt: Instant) {
fun isExpired() = expiresAt.isAfter(Instant.now().minusSeconds(60L))

Check warning on line 134 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L133-L134

Added lines #L133 - L134 were not covered by tests
}

fun readPrivateKey(filename: String): PrivateKey {
val pemParser = PEMParser(FileReader(File(filename)))
val o: PEMKeyPair = pemParser.readObject() as PEMKeyPair
val converter = JcaPEMKeyConverter().setProvider("BC")
val kp = converter.getKeyPair(o)
return kp.private

Check warning on line 142 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L138-L142

Added lines #L138 - L142 were not covered by tests
}

fun createJWT(githubAppId: String, ttlMillis: Long, privateKey: PrivateKey): Jwt {
//The JWT signature algorithm we will be using to sign the token
val signatureAlgorithm: SecureDigestAlgorithm<PrivateKey, PublicKey> = Jwts.SIG.RS256

Check warning on line 147 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L147

Added line #L147 was not covered by tests

val nowMillis = System.currentTimeMillis()
val now = Date(nowMillis)

Check warning on line 150 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L149-L150

Added lines #L149 - L150 were not covered by tests

//Let's set the JWT Claims
val builder: JwtBuilder = Jwts.builder()
.issuedAt(now)
.issuer(githubAppId)
.signWith(privateKey, signatureAlgorithm)

Check warning on line 156 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L153-L156

Added lines #L153 - L156 were not covered by tests

//if it has been specified, let's add the expiration
if (ttlMillis > 0) {
val expMillis = nowMillis + ttlMillis
val exp = Date(expMillis)
builder.expiration(exp)

Check warning on line 162 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L160-L162

Added lines #L160 - L162 were not covered by tests
}

//Builds the JWT and serializes it to a compact, URL-safe string
return Jwt(builder.compact(), nowMillis, ttlMillis)

Check warning on line 166 in src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt

View check run for this annotation

Codecov / codecov/patch

src/main/kotlin/com/github/lernejo/korekto/toolkit/thirdparty/github/GitHubAuthenticationHolder.kt#L166

Added line #L166 was not covered by tests
}
Loading

0 comments on commit b6d1855

Please sign in to comment.