firebase - 每次我尝试调用云函数时,它都会通过异常代码“NOT_FOUND”
问题描述
描述
我正在使用 firebase 新更新服务https://firebase.google.com/docs/ml/android/recognize-text的错误。我正在尝试从图像中打印文本数据,但异常通过。.
重现
这是我的 kotlin 代码
package jp.co.camrininfotech.mlText
import android.Manifest
import android.app.Activity
import android.content.Intent
import android.content.pm.PackageManager
import android.graphics.Bitmap
import android.net.Uri
import android.os.Bundle
import android.os.Environment
import android.provider.MediaStore
import android.provider.Settings
import android.util.Base64
import android.util.Log
import android.view.View
import android.widget.ImageView
import android.widget.TextView
import androidx.appcompat.app.AlertDialog
import androidx.appcompat.app.AppCompatActivity
import androidx.core.app.ActivityCompat
import com.google.android.gms.tasks.Task
import com.google.android.material.snackbar.Snackbar
import com.google.firebase.functions.FirebaseFunctions
import com.google.firebase.functions.FirebaseFunctionsException
import com.google.firebase.functions.ktx.functions
import com.google.firebase.ktx.Firebase
import com.google.gson.*
import java.io.ByteArrayOutputStream
import java.io.File
class MainActivity : AppCompatActivity(), View.OnClickListener {
private lateinit var functions: FirebaseFunctions
private var myImageView: ImageView? = null
private var myTextView: TextView? = null
private var WRITE_STORAGE = 100
private var SELECT_PHOTO = 102
private val request = JsonObject()
var photo: File? = null
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
myTextView = findViewById(R.id.textView)
myImageView = findViewById(R.id.imageView)
findViewById<View>(R.id.checkText).setOnClickListener(this)
findViewById<View>(R.id.select_image).setOnClickListener(this)
functions = Firebase.functions
}
override fun onClick(view: View?) {
if (view != null) {
when (view.getId()) {
R.id.checkText -> {
processImage()
}
R.id.select_image -> selectPicture()
}
}
}
override fun onRequestPermissionsResult(
requestCode: Int,
permissions: Array<String?>,
grantResults: IntArray
) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
when (requestCode) {
WRITE_STORAGE ->
if (grantResults.size > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
selectPicture()
} else {
requestPermission(this, requestCode, R.string.permission_request)
}
}
}
private fun requestPermission(activity: Activity, requestCode: Int, msg: Int) {
val alert = AlertDialog.Builder(activity)
alert.setMessage(msg)
alert.setPositiveButton(
android.R.string.ok
) { dialogInterface, i ->
dialogInterface.dismiss()
val permissonIntent = Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS)
permissonIntent.data = Uri.parse("package:" + activity.packageName)
activity.startActivityForResult(permissonIntent, requestCode)
}
alert.setNegativeButton(
android.R.string.cancel
) { dialogInterface, i -> dialogInterface.dismiss() }
alert.setCancelable(false)
alert.show()
}
private fun selectPicture() {
photo = createTempFile(photo)
val intent = Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI)
startActivityForResult(intent, SELECT_PHOTO)
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
val dataUri = data!!.data
var bitmap: Bitmap = MediaStore.Images.Media.getBitmap(contentResolver, dataUri)
// Scale down bitmap size
bitmap = scaleBitmapDown(bitmap, 640)
myImageView!!.setImageBitmap(bitmap)
convertIntoBase(bitmap)
}
private fun scaleBitmapDown(bitmap: Bitmap, maxDimension: Int): Bitmap {
val originalWidth = bitmap.width
val originalHeight = bitmap.height
var resizedWidth = maxDimension
var resizedHeight = maxDimension
if (originalHeight > originalWidth) {
resizedHeight = maxDimension
resizedWidth =
(resizedHeight * originalWidth.toFloat() / originalHeight.toFloat()).toInt()
} else if (originalWidth > originalHeight) {
resizedWidth = maxDimension
resizedHeight =
(resizedWidth * originalHeight.toFloat() / originalWidth.toFloat()).toInt()
} else if (originalHeight == originalWidth) {
resizedHeight = maxDimension
resizedWidth = maxDimension
}
return Bitmap.createScaledBitmap(bitmap, resizedWidth, resizedHeight, false)
}
private fun convertIntoBase(bitmap: Bitmap) {
val byteArrayOutputStream = ByteArrayOutputStream()
bitmap.compress(Bitmap.CompressFormat.JPEG, 100, byteArrayOutputStream)
val imageBytes: ByteArray = byteArrayOutputStream.toByteArray()
val base64encoded = Base64.encodeToString(imageBytes, Base64.NO_WRAP)
requestJsonToCloud(base64encoded)
}
private fun requestJsonToCloud(base64encoded: String) {
val image = JsonObject()
image.add("content", JsonPrimitive(base64encoded))
request.add("image", image)
val feature = JsonObject()
feature.add("type", JsonPrimitive("TEXT_DETECTION"))
val features = JsonArray()
features.add(feature)
request.add("features", features)
val imageContext = JsonObject()
val languageHints = JsonArray()
languageHints.add("en")
imageContext.add("languageHints", languageHints)
request.add("imageContext", imageContext)
}
private fun processImage() {
annotateImage(request.toString())
.addOnCompleteListener { task ->
if (!task.isSuccessful) {
val e = task.exception
if (e is FirebaseFunctionsException) {
val code = e.code
val details = e.details
Log.w("exception code ", "$code")
Log.w("exception details", "$details")
}
showSnackbar("An error occurred.")
} else {
val annotation =
task.result!!.asJsonArray[0].asJsonObject["fullTextAnnotation"].asJsonObject
System.out.format("%nComplete annotation:")
System.out.format("%n%s", annotation["text"].asString)
}
}
}
private fun showSnackbar(message: String) {
Snackbar.make(findViewById(android.R.id.content), message, Snackbar.LENGTH_SHORT).show()
}
private fun createTempFile(fileClass: File?): File {
var file = fileClass
val directory =
File(Environment.getExternalStorageDirectory().path + "/jp.co.camrininfotech.mlText")
if (!directory.exists() || !directory.isDirectory) {
directory.mkdirs()
}
if (file == null) {
file = File(directory, "orig.jpg")
}
return file
}
private fun annotateImage(requestJson: String): Task<JsonElement> {
return functions
.getHttpsCallable("annotateImage")
.call(requestJson)
.continueWith { task ->
println("annotateImage result " + task.result.toString())
val result = task.result?.data
JsonParser.parseString(Gson().toJson(result))
}
}
private fun checkPermission(requestCode: Int) {
when (requestCode) {
WRITE_STORAGE -> {
val hasWriteExternalStoragePermission = ActivityCompat.checkSelfPermission(
this,
Manifest.permission.WRITE_EXTERNAL_STORAGE
)
if (hasWriteExternalStoragePermission == PackageManager.PERMISSION_GRANTED) {
selectPicture()
} else {
ActivityCompat.requestPermissions(
this,
arrayOf(Manifest.permission.WRITE_EXTERNAL_STORAGE),
requestCode
)
}
}
}
}
}
我正在使用这些依赖项
implementation 'com.google.firebase:firebase-auth-ktx'
implementation platform('com.google.firebase:firebase-bom:28.0.1')
implementation 'com.google.android.gms:play-services-auth:19.0.0'
implementation 'com.firebaseui:firebase-ui-auth:7.1.1'
implementation 'com.google.firebase:firebase-functions-ktx'
implementation 'com.google.firebase:firebase-analytics-ktx'
预期行为
从图像中打印文本
附加上下文
W/异常代码:NOT_FOUND W/异常详细信息:null
解决方案
我得到了这个问题的解决方案。我没有在我的 firebase 云功能中上传vision-annotate-images 。
To deploy the function:
Clone or download the functions-samples repo and change to the vision-annotate-image directory:
git clone https://github.com/firebase/functions-samples
cd vision-annotate-image
Install dependencies:
cd functions
npm install
cd ..
If you don't have the Firebase CLI, install it.
Initialize a Firebase project in the vision-annotate-image directory. When prompted, select your project in the list.
firebase init
Deploy the function:
firebase deploy --only functions:annotateImage
推荐阅读
- c - c语言密码检查器
- powerbi-embedded - 0x800a138f - JavaScript 运行时错误:无法在 powerbi.js 和 powerbi.min.js 中获取未定义或空引用的属性“getRandomValues”
- javascript - 从 Firestore 读取的谷歌云函数的变化无常的执行时间
- c++ - 为什么 `clang++ -fuse-ld=lld main.cpp` 不起作用?
- java - 单击电子邮件中的链接时如何生成不记名令牌
- c# - 在 UWP 中使用 RatingControl 时如何让用户选择半星?
- angular - 按键处理程序慢
- go - go-twitter API 提取视频 URL
- django - S3下载后在Django中显示成功消息
- android - 使用 Firebase AuthUI 进行身份验证的 Android 动态功能模块