Skip to content

Commit

Permalink
Fix JNI conversion for key names
Browse files Browse the repository at this point in the history
  • Loading branch information
Benjamin AIMONE authored and Benjamin AIMONE committed Sep 28, 2022
1 parent 4023839 commit 85ea7dc
Show file tree
Hide file tree
Showing 8 changed files with 131 additions and 15 deletions.
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
# 0.2.1
### Fixes
* Convert key names to C++ style to avoid `No implementation found` error
#### Contributors
@pratclot
# 0.2.0
### Improvements
* Ability to re-generate secrets from .properties file
Expand Down
7 changes: 4 additions & 3 deletions build.gradle.kts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
plugins {
id("com.gradle.plugin-publish") version "0.21.0"
id("com.gradle.plugin-publish") version "1.0.0"
id("io.gitlab.arturbosch.detekt") version "1.19.0"
`kotlin-dsl`
`maven-publish`
Expand All @@ -14,8 +14,9 @@ repositories {
dependencies {
implementation("com.android.tools.build:gradle:4.2.2")

testImplementation("io.kotest:kotest-runner-junit5-jvm:5.2.2")
testImplementation("io.kotest:kotest-assertions-core-jvm:5.2.2")
testImplementation("io.kotest:kotest-runner-junit5-jvm:5.4.2")
testImplementation("io.kotest:kotest-assertions-core-jvm:5.4.2")
testImplementation("io.kotest:kotest-framework-datatest-jvm:5.4.2")
testImplementation("junit:junit:4.13.2")
}

Expand Down
2 changes: 1 addition & 1 deletion settings.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -2,5 +2,5 @@ rootProject.name = "HiddenSecretsPlugin"

gradle.allprojects {
group = "com.klaxit.hiddensecrets"
version = "0.2.0"
version = "0.2.1"
}
2 changes: 1 addition & 1 deletion src/main/kotlin/com/klaxit/hiddensecrets/CodeGenerator.kt
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ object CodeGenerator {

return "\nextern \"C\"\n" +
"JNIEXPORT jstring JNICALL\n" +
"Java_" + Utils.getCppPackageName(packageName) + "_Secrets_get$keyName(\n" +
"Java_" + Utils.getCppName(packageName) + "_Secrets_get$keyName(\n" +
" JNIEnv* pEnv,\n" +
" jobject pThis,\n" +
" jstring packageName) {\n" +
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -295,18 +295,20 @@ open class HiddenSecretsPlugin : Plugin<Project> {
if (text.contains(obfuscatedKey)) {
println("⚠️ Key already added in C++ !")
}
// Escape required characters
val cppKeyName = Utils.getCppName(keyName)
if (text.contains(KEY_PLACEHOLDER)) {
// Edit placeholder key
// Replace package name
text = text.replace(PACKAGE_PLACEHOLDER, Utils.getCppPackageName(kotlinPackage))
text = text.replace(PACKAGE_PLACEHOLDER, Utils.getCppName(kotlinPackage))
// Replace key name
text = text.replace("YOUR_KEY_NAME_GOES_HERE", keyName)
text = text.replace("YOUR_KEY_NAME_GOES_HERE", cppKeyName)
// Replace demo key
text = text.replace(KEY_PLACEHOLDER, obfuscatedKey)
secretsCpp.writeText(text)
} else {
// Add new key
text += CodeGenerator.getCppCode(kotlinPackage, keyName, obfuscatedKey)
text += CodeGenerator.getCppCode(kotlinPackage, cppKeyName, obfuscatedKey)
secretsCpp.writeText(text)
}
} else {
Expand Down
6 changes: 3 additions & 3 deletions src/main/kotlin/com/klaxit/hiddensecrets/Utils.kt
Original file line number Diff line number Diff line change
Expand Up @@ -10,11 +10,11 @@ import kotlin.experimental.xor
object Utils {

/**
* Transform package name com.klaxit.hidden to com_klaxit_hidden to ingrate in C++ code
* Java package name needs to escape some characters to call the NDK
* Transform names like com.klaxit.hidden to com_klaxit_hidden to ingrate in C++ code.
* Java package name and key names needs to escape some characters to call the NDK.
* From https://docs.oracle.com/javase/8/docs/technotes/guides/jni/spec/design.html#resolving_native_method_names
*/
fun getCppPackageName(packageName: String): String {
fun getCppName(packageName: String): String {
return packageName
.replace("_", "_1")
.replace(";", "_2")
Expand Down
108 changes: 108 additions & 0 deletions src/test/kotlin/IntegrationTest.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
import com.klaxit.hiddensecrets.HiddenSecretsPlugin
import io.kotest.core.spec.style.WordSpec
import io.kotest.data.Row4
import io.kotest.datatest.withData
import io.kotest.matchers.shouldBe
import org.gradle.testkit.runner.GradleRunner
import org.junit.rules.TemporaryFolder
import java.io.File

/**
* Test that correct names of keys are present in java and cpp files.
*
* From @pratclot
* https://github.com/klaxit/hidden-secrets-gradle-plugin/pull/64
*/
class IntegrationTest : WordSpec({

"Apply the plugin" should {
val testProjectDir = TemporaryFolder()
testProjectDir.create()
val buildFile = testProjectDir.newFile("build.gradle")
buildFile.appendText(
"""
plugins {
id 'com.klaxit.hiddensecrets'
id 'com.android.application'
}
android {
compileSdkVersion 29
}
""".trimIndent()
)

val gradleRunner = GradleRunner.create()
.withPluginClasspath()
.withProjectDir(testProjectDir.root)
.withTestKitDir(testProjectDir.newFolder())

val test = this
"Make command ${HiddenSecretsPlugin.TASK_HIDE_SECRET} succeed" {
test.withData(
Row4("thisIsATestKey", "thisIsATestKeyName", "thisIsATestKeyName", "com.package.test"),
Row4(
"this_is_a_test_key",
"this_is_a_test_key_name",
"this_1is_1a_1test_1key_1name",
"com.package.test"
),
) { (key, keyName, cppKeyName, packageName) ->
testProjectDir.run {
val packagePath = packageName.replace('.', '/')
val packageDirJava = newFolder("src/main/java/$packagePath")
val packageDirCpp = newFolder("src/main/cpp/")

val fileJava = File(packageDirJava, "Secrets.kt")
val fileCpp = File(packageDirCpp, "secrets.cpp")

var inputStream = javaClass.classLoader.getResourceAsStream("kotlin/Secrets.kt")
inputStream?.bufferedReader()?.lines()?.forEach {
fileJava.appendText(it + "\n")
}
inputStream?.close()

inputStream = javaClass.classLoader.getResourceAsStream("cpp/secrets.cpp")
inputStream?.bufferedReader()?.forEachLine {
fileCpp.appendText(it + "\n")
}
inputStream?.close()

gradleRunner.withArguments()
val result = gradleRunner
.withArguments(
HiddenSecretsPlugin.TASK_HIDE_SECRET,
"-Pkey=$key",
"-PkeyName=$keyName",
"-Ppackage=$packageName"
)
.build()
println(result.output)

// Search key name in java
var javaCorrectNameFound = false
fileJava.bufferedReader().forEachLine {
if (it.contains(keyName)) {
javaCorrectNameFound = true
}
}
javaCorrectNameFound shouldBe true
// Search function name in ccp
val expectedCppFunctionName = "Java_${packageName.replace('.', '_')}_Secrets_get$cppKeyName("
println("expectedFunctionName: $expectedCppFunctionName")
var cppCorrectNameFound = false
fileCpp.bufferedReader().forEachLine {
if (it.contains(expectedCppFunctionName)) {
cppCorrectNameFound = true
}
}
cppCorrectNameFound shouldBe true

fileJava.delete()
fileCpp.delete()
packageDirJava.delete()
packageDirCpp.delete()
}
}
}
}
})
8 changes: 4 additions & 4 deletions src/test/kotlin/UtilsTest.kt
Original file line number Diff line number Diff line change
Expand Up @@ -10,15 +10,15 @@ class UtilsTest : WordSpec({

val packageName = "com.klaxit.test"

"Using getCppPackageName()" should {
"Using getCppName()" should {
"transform package separator" {
Utils.getCppPackageName(packageName) shouldBe "com_klaxit_test"
Utils.getCppName(packageName) shouldBe "com_klaxit_test"
}
"transform package with underscore" {
Utils.getCppPackageName("com.klaxit.test_with_underscore") shouldBe "com_klaxit_test_1with_1underscore"
Utils.getCppName("com.klaxit.test_with_underscore") shouldBe "com_klaxit_test_1with_1underscore"
}
"transform package with escaping characters" {
Utils.getCppPackageName("com[test.klaxit;test.test_with_underscore") shouldBe "com_3test_klaxit_2test_test_1with_1underscore"
Utils.getCppName("com[test.klaxit;test.test_with_underscore") shouldBe "com_3test_klaxit_2test_test_1with_1underscore"
}
}

Expand Down

0 comments on commit 85ea7dc

Please sign in to comment.