Как загрузить файл с помощью сервиса - PullRequest
0 голосов
/ 25 августа 2018

Моя цель состоит в том, чтобы загружать файлы с службой (пожалуйста, вы можете предложить лучший вариант) в фоновом режиме без необходимости взаимодействия с приложением, чтобы можно было инициировать загрузку ( startService) всякий раз, когда он получает сообщение о том, что новый текст был скопирован в буфер обмена, и текст имеет загружаемое расширение.

В приведенной ниже реализации используется PR Downloader (Библиотека) для загрузкифайл, но файл не загружен.

Обычно я получаю сообщение об ошибке, и я регистрирую ответ.

onError. isServerError = false. isConnectionError = true

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest
xmlns:android="http://schemas.android.com/apk/res/android"
package="com.package.name">
    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

    <application
        ...>

        <activity
            android:name=".Activities.MainActivity"
            android:label="@string/app_name"
            android:theme="@style/AppTheme">
            <intent-filter>
                ...
            </intent-filter>
        </activity>

        ...
        <service android:name=".DownloadService" />
    </application>
</manifest>

MainActivity.kt

class MainActivity : AppCompatActivity() {
        override fun onCreate(savedInstanceState: Bundle?) {
            super.onCreate(savedInstanceState)
            setContentView(R.layout.activity_main)
            val clipBoard = getSystemService(Context.CLIPBOARD_SERVICE) as ClipboardManager
            clipBoard.addPrimaryClipChangedListener {
                val primaryClip = clipBoard.primaryClip

                if (primaryClip == null || primaryClip.itemCount > 0 && primaryClip.getItemAt(0).text == null)
                    return@addPrimaryClipChangedListener  
                val serviceIntent = Intent(this, DownloadService::class.java)
                val clip = primaryClip.getItemAt(0).text.toString()
                if (clip.contains(".mp3")) {
                    Log.d("MainActivity","$clip")
                    serviceIntent.putExtra(DOWNLOAD_URL_KEY, clip)
                    serviceIntent.putExtra(DOWNLOAD_FILE_NAME_KEY, getFileName(clip))
                    startService(serviceIntent)
                }
            }
        }
    }

DownloadService.kt

class DownloadService : Service() {
    override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
        val url = intent?.getStringExtra(DOWNLOAD_URL_KEY)
        val fileName = intent?.getStringExtra(DOWNLOAD_FILE_NAME_KEY)
        Log("onStartCommand. FileName = $fileName Download URL = $url")

        if (url != null)
            downoadFile(url, fileName)

        return super.onStartCommand(intent, flags, startId)
    }
    fun downoadFile(download_url: String, fileName: String?) {
        val dirPath ="/Music Downloader/$fileName.mp3"
        val downloadId = PRDownloader.download(download_url, dirPath, fileName)
                .build()
                .setOnStartOrResumeListener {

                }
                .setOnPauseListener {}
                .setOnCancelListener {}
                .setOnProgressListener { progress ->

                    Log("Progress = ${((progress.currentBytes / progress.totalBytes) * 100) as Int}%")
                }
                .start(object : OnDownloadListener {
                    override fun onError(error: com.downloader.Error?) {
                        Log("onError. ServerError = ${error?.isServerError}. ConnectionError = ${error?.isConnectionError}")
                    }

                    override fun onDownloadComplete() {
                        Log("onDownloadComplete")
                    }
                })
        Log("dirPath = $dirPath, download_url = $download_url, Download ID = $downloadId")

    }
}

Я также пытался использовать Fetch но тот же результат - файл не загружается.

Но я заметил, что получение и анализ файла JSON / Xml работают с VOLLEY (:) Я знаю, что это 2018. Я перейду на Retrofit, обещаю).

Пожалуйста, что я делаю не так?Как я могу загрузить файл с помощью службы.

ОБНОВЛЕНИЕ

Я попытался запустить код в основной деятельности, но он все еще не работал.Я тестирую с Android 8.1, и я думаю, что-то не так с библиотекой PR-Downloader

override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        PRDownloader.initialize(getApplicationContext())
        val download_url = "https://images.pexels.com/photos/658687/pexels-photo-658687.jpeg?cs=srgb&dl=beautiful-bloom-blooming-658687.jpg"
      download(download_url)
}
fun downoad(url: String, fileName: String) {
        val dirPath = "/File Downloader/" + fileName
        val downloadId = PRDownloader.download(url, dirPath, fileName)
                .build()
                .setOnStartOrResumeListener {

                }
                .setOnPauseListener {}
                .setOnCancelListener {}
                .setOnProgressListener { progress ->

                    Log("Progress = ${((progress.currentBytes / progress.totalBytes) * 100) as Int}%")
                }
                .start(object : OnDownloadListener {
                    override fun onError(error: com.downloader.Error?) {
                        Log("onError. isServerError = ${error?.isServerError}. isConnectionError = ${error?.isConnectionError}")
                    }

                    override fun onDownloadComplete() {
                        Log("onDownloadComplete")
                    }
                })
    }

Ответы [ 2 ]

0 голосов
/ 27 августа 2018

Используйте класс DownloadManager:

private var dm: DownloadManager? = null
companion object {
        lateinit var receiver: BroadcastReceiver
    }

вызовите этот метод внутри вашего onCreate:

private fun initbroadCastReceiver() {

        receiver = object : BroadcastReceiver() {
            override fun onReceive(context: Context, intent: Intent) {
                val action = intent.action
                if (DownloadManager.ACTION_DOWNLOAD_COMPLETE == action) {
                    val downloadId = intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID, 0)
                    val query = Query()
                    query.setFilterById(enqueue)
                    val c = dm!!.query(query)
                    if (c.moveToFirst()) {
                        val columnIndex = c.getColumnIndex(DownloadManager.COLUMN_STATUS)
                        if (DownloadManager.STATUS_SUCCESSFUL == c.getInt(columnIndex)) {
                            isDownload = true
                            downloadImage.setImageResource(R.drawable.icon_download_grey)
                            downloadImage.setBackground(ContextCompat.getDrawable(baseActivity, R.drawable.circle_background_grey))
                            SQLHelper.updateData(bookEntity.id)
                            showToast("Download completed")
                        }
                    }
                }
            }
        }
    }

onClick for Download:

downloadImage.setOnClickListener(View.OnClickListener
        {
            if (bookEntity.isDownload == 0 && isDownload == false) {
                if (Misc.isNetworkAvailable(baseActivity)) {
                    dm = baseActivity.getSystemService(DOWNLOAD_SERVICE) as DownloadManager
                    val request = Request(
                            Uri.parse(EnvironmentConstant.BOOK_DOWNLOAD_URL + bookEntity.id))
                            .setTitle(""Title)
                            .setDestinationInExternalFilesDir(baseActivity,path)
                            .setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED)
                    enqueue = dm!!.enqueue(request)
                } else {
                    showAlertDialog()
                }
            } else {
                alreadyDownloadAlert()
            }
        })
0 голосов
/ 25 августа 2018

Оказалось, проблема в том, что ваш download_url , похоже, "images.pexels.com" не разрешает подключения к своим фотографиям. Я протестировал приведенный ниже код (который является загрузчиком на чистом Java, который я реализовал) и получил исключения Connection.refused. Попробуйте свой собственный код с https://html.com/wp-content/uploads/flamingo.jpg или https://upload.wikimedia.org/wikipedia/commons/d/de/Bananavarieties.jpg и посмотрите, что вы получите

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_main)
    download().execute("https://images.pexels.com/photos/658687/pexels-photo-658687.jpeg",
            Environment.getExternalStorageDirectory().path+"/myFolder/", "pexels-photo-658687.jpeg")
}

class download : AsyncTask<String, String, String>() {
    override fun doInBackground(vararg p0: String?): String? {
        val url = URL(p0[0])
        val buf = ByteArray(1024)
        var byteRead: Int
        val p = File(p0[1])
        p.mkdirs()
        val f = File(p0[1], p0[2])
        f.createNewFile()
        var outStream = BufferedOutputStream(
                FileOutputStream(p0[1] + p0[2]))

        val conn = url.openConnection()
        var `is` = conn.getInputStream()
        byteRead = `is`!!.read(buf)
        while (byteRead != -1) {
            outStream.write(buf, 0, byteRead)
            byteRead = `is`.read(buf)
        }
        `is`.close()
        outStream.close()
        return null;
    }
}
...