summaryrefslogtreecommitdiff
path: root/PermissionController/src/com/android/permissioncontroller/DumpableLog.kt
blob: 56682d018f05839be520131139e9fe80967ccb66 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
/*
 * Copyright (C) 2020 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.android.permissioncontroller

import android.util.Log
import com.android.permissioncontroller.Constants.LOGS_TO_DUMP_FILE
import java.io.File

/** Like {@link Log} but stores the logs in a file which can later be dumped via {@link #dump} */
object DumpableLog {
    private const val MAX_FILE_SIZE = 64 * 1024

    private val lock = Any()
    private val file = File(PermissionControllerApplication.get().filesDir, LOGS_TO_DUMP_FILE)

    init {
        file.createNewFile()
    }

    /** Equivalent to {@link Log.v} */
    fun v(tag: String, message: String, exception: Throwable? = null) {
        Log.v(tag, message, exception)
        addLogToDump("v", tag, message, exception)
    }

    /** Equivalent to {@link Log.d} */
    fun d(tag: String, message: String, exception: Throwable? = null) {
        Log.d(tag, message, exception)
        addLogToDump("d", tag, message, exception)
    }

    /** Equivalent to {@link Log.i} */
    fun i(tag: String, message: String, exception: Throwable? = null) {
        Log.i(tag, message, exception)
        addLogToDump("i", tag, message, exception)
    }

    /** Equivalent to {@link Log.w} */
    fun w(tag: String, message: String, exception: Throwable? = null) {
        Log.w(tag, message, exception)
        addLogToDump("w", tag, message, exception)
    }

    /** Equivalent to {@link Log.e} */
    fun e(tag: String, message: String, exception: Throwable? = null) {
        Log.e(tag, message, exception)
        addLogToDump("e", tag, message, exception)
    }

    private fun addLogToDump(level: String, tag: String, message: String, exception: Throwable?) {
        synchronized(lock) {
            // TODO: Needs to be replaced by proper log rotation
            if (file.length() > MAX_FILE_SIZE) {
                val dump = file.readLines()

                file.writeText("truncated at ${System.currentTimeMillis()}\n")
                dump.subList(dump.size / 2, dump.size).forEach { file.appendText(it + "\n") }
            }

            file.appendText(
                "${System.currentTimeMillis()} $tag:$level $message " +
                    "${exception?.let { it.message + Log.getStackTraceString(it) } ?: ""}\n"
            )
        }
    }

    /** @return the previously logged entries */
    suspend fun get(): List<String> {
        synchronized(lock) {
            return file.readLines()
        }
    }
}