All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jetbrains.kotlin.daemon.common.experimental.SecurityUtils.kt Maven / Gradle / Ivy

There is a newer version: 2.0.20-RC
Show newest version
/*
 * Copyright 2010-2018 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license
 * that can be found in the license/LICENSE.txt file.
 */

@file:Suppress("CAST_NEVER_SUCCEEDS")

package org.jetbrains.kotlin.daemon.common.experimental

import org.jetbrains.kotlin.daemon.common.experimental.socketInfrastructure.ByteReadChannelWrapper
import org.jetbrains.kotlin.daemon.common.experimental.socketInfrastructure.ByteWriteChannelWrapper
import java.io.FileInputStream
import java.io.FileOutputStream
import java.io.ObjectInputStream
import java.io.ObjectOutputStream
import java.security.*
import java.security.spec.InvalidKeySpecException
import java.security.spec.X509EncodedKeySpec


const val SECURITY_TOKEN_SIZE = 128

private val secureRandom = SecureRandom.getInstance("SHA1PRNG", "SUN");
private val pairGenerator = KeyPairGenerator.getInstance("DSA", "SUN")

private fun generateSecurityToken(): ByteArray {
    val tokenBuffer = ByteArray(SECURITY_TOKEN_SIZE)
    secureRandom.nextBytes(tokenBuffer)
    return tokenBuffer
}

data class SecurityData(val privateKey: PrivateKey, val publicKey: PublicKey, val token: ByteArray)
fun generateKeysAndToken() = pairGenerator.generateKeyPair().let {
    SecurityData(it.private, it.public, generateSecurityToken())
}

private fun FileInputStream.readBytesFixedLength(n: Int): ByteArray {
    val buffer = ByteArray(n)
    var bytesRead = 0
    while (bytesRead != n) {
        bytesRead += this.read(buffer, bytesRead, n - bytesRead)
    }
    return buffer
}

// server part :
fun sendTokenKeyPair(output: FileOutputStream, token: ByteArray, privateKey: PrivateKey) {
    output.write(token)
    ObjectOutputStream(output).use {
        it.writeObject(privateKey)
    }
}

private fun instantiateDsa() = Signature.getInstance("SHA1withDSA", "SUN")

suspend fun getSignatureAndVerify(input: ByteReadChannelWrapper, expectedToken: ByteArray, publicKey: PublicKey): Boolean {
    val signature = input.nextBytes()
    val dsa = instantiateDsa()
    dsa.initVerify(publicKey)
    dsa.update(expectedToken, 0, SECURITY_TOKEN_SIZE)
    val verified = dsa.verify(signature)
    log.fine("verified : $verified")
    return verified
}


// client part :
fun readTokenKeyPairAndSign(input: FileInputStream): ByteArray {
    val token = input.readBytesFixedLength(SECURITY_TOKEN_SIZE)
    val privateKey = ObjectInputStream(input).use(ObjectInputStream::readObject) as PrivateKey
    val dsa = instantiateDsa()
    dsa.initSign(privateKey)
    dsa.update(token, 0, SECURITY_TOKEN_SIZE)
    return dsa.sign()
}

suspend fun sendSignature(output: ByteWriteChannelWrapper, signature: ByteArray) = output.writeBytesAndLength(signature.size, signature)




© 2015 - 2024 Weber Informatics LLC | Privacy Policy