Skip to content

Commit

Permalink
Users can export unencrypted data to JSON or CSV
Browse files Browse the repository at this point in the history
  • Loading branch information
SaintPatrck committed Apr 22, 2024
1 parent 1686ae0 commit 69da880
Show file tree
Hide file tree
Showing 23 changed files with 782 additions and 12 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
package com.bitwarden.authenticator.data.authenticator.manager

import android.net.Uri

/**
* Manages reading and writing files.
*/
interface FileManager {

/**
* Writes the given [dataString] to disk at the provided [fileUri]
*/
suspend fun stringToUri(fileUri: Uri, dataString: String): Boolean
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
package com.bitwarden.authenticator.data.authenticator.manager

import android.content.Context
import android.net.Uri
import com.bitwarden.authenticator.data.platform.manager.DispatcherManager
import kotlinx.coroutines.withContext

class FileManagerImpl(
private val context: Context,
private val dispatcherManager: DispatcherManager,
) : FileManager {

override suspend fun stringToUri(fileUri: Uri, dataString: String): Boolean {
@Suppress("TooGenericExceptionCaught")
return try {
withContext(dispatcherManager.io) {
context
.contentResolver
.openOutputStream(fileUri)
?.use { outputStream ->
outputStream.write(dataString.toByteArray())
}
}
true
} catch (exception: RuntimeException) {
false
}
}
}
Original file line number Diff line number Diff line change
@@ -1,12 +1,16 @@
package com.bitwarden.authenticator.data.authenticator.manager.di

import android.content.Context
import com.bitwarden.authenticator.data.authenticator.datasource.sdk.AuthenticatorSdkSource
import com.bitwarden.authenticator.data.authenticator.manager.FileManager
import com.bitwarden.authenticator.data.authenticator.manager.FileManagerImpl
import com.bitwarden.authenticator.data.authenticator.manager.TotpCodeManager
import com.bitwarden.authenticator.data.authenticator.manager.TotpCodeManagerImpl
import com.bitwarden.authenticator.data.platform.manager.DispatcherManager
import dagger.Module
import dagger.Provides
import dagger.hilt.InstallIn
import dagger.hilt.android.qualifiers.ApplicationContext
import dagger.hilt.components.SingletonComponent
import java.time.Clock
import javax.inject.Singleton
Expand All @@ -18,6 +22,16 @@ import javax.inject.Singleton
@InstallIn(SingletonComponent::class)
object AuthenticatorManagerModule {

@Provides
@Singleton
fun providerFileManager(
@ApplicationContext context: Context,
dispatcherManager: DispatcherManager,
) : FileManager = FileManagerImpl(
context = context,
dispatcherManager = dispatcherManager,
)

@Provides
@Singleton
fun provideTotpCodeManager(
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
package com.bitwarden.authenticator.data.authenticator.manager.model

import kotlinx.serialization.Serializable

/**
* Models exported authenticator data in JSON format.
*
* This model is loosely based off of Bitwarden's exported unencrypted vault data.
*/
@Serializable
data class ExportJsonData(
val encrypted: Boolean,
val items: List<ExportItem>,
) {

/**
* Represents a single exported authenticator item.
*
* This model is loosely based off of Bitwarden's exported Cipher JSON.
*/
@Serializable
data class ExportItem(
val id: String,
val name: String,
val folderId: String?,
val organizationId: String?,
val collectionIds: List<String>?,
val notes: String?,
val type: Int,
val login: ItemLoginData,
val favorite: Boolean,
) {
/**
* Represents the login specific data of an exported item.
*
* This model is loosely based off of Bitwarden's Cipher.Login JSON.
*
* @property totp OTP secret used to generate a verification code.
* @property issuer Optional issuer of the 2fa code.
* @property period Optional refresh period in seconds. Default is 30.
* @property digits Optional number of digits in the verification code. Default is 6
*/
@Serializable
data class ItemLoginData(
val totp: String,
val issuer: String?,
val period: Int,
val digits: Int,
)
}
}
Original file line number Diff line number Diff line change
@@ -1,14 +1,17 @@
package com.bitwarden.authenticator.data.authenticator.repository

import android.net.Uri
import com.bitwarden.authenticator.data.authenticator.datasource.disk.entity.AuthenticatorItemEntity
import com.bitwarden.authenticator.data.authenticator.manager.model.VerificationCodeItem
import com.bitwarden.authenticator.data.authenticator.repository.model.AuthenticatorData
import com.bitwarden.authenticator.data.authenticator.repository.model.CreateItemResult
import com.bitwarden.authenticator.data.authenticator.repository.model.DeleteItemResult
import com.bitwarden.authenticator.data.authenticator.repository.model.ExportDataResult
import com.bitwarden.authenticator.data.authenticator.repository.model.TotpCodeResult
import com.bitwarden.authenticator.data.authenticator.repository.model.UpdateItemRequest
import com.bitwarden.authenticator.data.authenticator.repository.model.UpdateItemResult
import com.bitwarden.authenticator.data.platform.repository.model.DataState
import com.bitwarden.authenticator.ui.platform.feature.settings.export.model.ExportFormat
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.StateFlow

Expand Down Expand Up @@ -78,4 +81,9 @@ interface AuthenticatorRepository {
itemId: String,
updateItemRequest: UpdateItemRequest,
): UpdateItemResult

/**
* Attempt to get the user's data for export.
*/
suspend fun exportVaultData(format: ExportFormat, fileUri: Uri): ExportDataResult
}
Original file line number Diff line number Diff line change
@@ -1,12 +1,16 @@
package com.bitwarden.authenticator.data.authenticator.repository

import android.net.Uri
import com.bitwarden.authenticator.data.authenticator.datasource.disk.AuthenticatorDiskSource
import com.bitwarden.authenticator.data.authenticator.datasource.disk.entity.AuthenticatorItemEntity
import com.bitwarden.authenticator.data.authenticator.manager.FileManager
import com.bitwarden.authenticator.data.authenticator.manager.TotpCodeManager
import com.bitwarden.authenticator.data.authenticator.manager.model.ExportJsonData
import com.bitwarden.authenticator.data.authenticator.manager.model.VerificationCodeItem
import com.bitwarden.authenticator.data.authenticator.repository.model.AuthenticatorData
import com.bitwarden.authenticator.data.authenticator.repository.model.CreateItemResult
import com.bitwarden.authenticator.data.authenticator.repository.model.DeleteItemResult
import com.bitwarden.authenticator.data.authenticator.repository.model.ExportDataResult
import com.bitwarden.authenticator.data.authenticator.repository.model.TotpCodeResult
import com.bitwarden.authenticator.data.authenticator.repository.model.UpdateItemRequest
import com.bitwarden.authenticator.data.authenticator.repository.model.UpdateItemResult
Expand All @@ -15,6 +19,7 @@ import com.bitwarden.authenticator.data.platform.repository.model.DataState
import com.bitwarden.authenticator.data.platform.repository.util.bufferedMutableSharedFlow
import com.bitwarden.authenticator.data.platform.repository.util.combineDataStates
import com.bitwarden.authenticator.data.platform.repository.util.map
import com.bitwarden.authenticator.ui.platform.feature.settings.export.model.ExportFormat
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.flow.Flow
Expand All @@ -23,13 +28,16 @@ import kotlinx.coroutines.flow.SharingStarted
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.asSharedFlow
import kotlinx.coroutines.flow.asStateFlow
import kotlinx.coroutines.flow.firstOrNull
import kotlinx.coroutines.flow.flatMapLatest
import kotlinx.coroutines.flow.flowOf
import kotlinx.coroutines.flow.launchIn
import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.onEach
import kotlinx.coroutines.flow.onStart
import kotlinx.coroutines.flow.stateIn
import kotlinx.serialization.encodeToString
import kotlinx.serialization.json.Json
import javax.inject.Inject

/**
Expand All @@ -41,6 +49,7 @@ private const val STOP_TIMEOUT_DELAY_MS: Long = 5_000L
class AuthenticatorRepositoryImpl @Inject constructor(
private val authenticatorDiskSource: AuthenticatorDiskSource,
private val totpCodeManager: TotpCodeManager,
private val fileManager: FileManager,
dispatcherManager: DispatcherManager,
) : AuthenticatorRepository {

Expand Down Expand Up @@ -214,4 +223,73 @@ class AuthenticatorRepositoryImpl @Inject constructor(
UpdateItemResult.Error(e.message)
}
}

override suspend fun exportVaultData(format: ExportFormat, fileUri: Uri): ExportDataResult {
return when (format) {
ExportFormat.JSON -> encodeVaultDataToJson(fileUri)
ExportFormat.CSV -> encodeVaultDataToCsv(fileUri)
}
}

private suspend fun encodeVaultDataToCsv(fileUri: Uri): ExportDataResult {
val headerLine =
"folder,favorite,type,name,login_uri,login_totp,issuer,period,digits"
val dataLines = authenticatorDiskSource
.getItems()
.firstOrNull()
.orEmpty()
.joinToString("\n") { it.toCsvFormat() }

val csvString = "$headerLine\n$dataLines"

return if (fileManager.stringToUri(fileUri = fileUri, dataString = csvString)) {
ExportDataResult.Success
} else {
ExportDataResult.Error
}
}

private fun AuthenticatorItemEntity.toCsvFormat() =
",,1,$accountName,,$key,$issuer,$period,$digits"

private suspend fun encodeVaultDataToJson(fileUri: Uri): ExportDataResult {
val dataString: String = Json.encodeToString(
ExportJsonData(
encrypted = false,
items = authenticatorDiskSource
.getItems()
.firstOrNull()
.orEmpty()
.map { it.toExportJsonItem() },
)
)

return if (
fileManager.stringToUri(
fileUri = fileUri,
dataString = dataString,
)
) {
ExportDataResult.Success
} else {
ExportDataResult.Error
}
}

private fun AuthenticatorItemEntity.toExportJsonItem() = ExportJsonData.ExportItem(
id = id,
folderId = null,
organizationId = null,
collectionIds = null,
name = accountName,
notes = null,
type = 1,
login = ExportJsonData.ExportItem.ItemLoginData(
totp = key,
issuer = issuer,
period = period,
digits = digits,
),
favorite = false,
)
}
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package com.bitwarden.authenticator.data.authenticator.repository.di

import com.bitwarden.authenticator.data.authenticator.datasource.disk.AuthenticatorDiskSource
import com.bitwarden.authenticator.data.authenticator.manager.FileManager
import com.bitwarden.authenticator.data.authenticator.manager.TotpCodeManager
import com.bitwarden.authenticator.data.authenticator.repository.AuthenticatorRepository
import com.bitwarden.authenticator.data.authenticator.repository.AuthenticatorRepositoryImpl
Expand All @@ -24,9 +25,11 @@ object AuthenticatorRepositoryModule {
authenticatorDiskSource: AuthenticatorDiskSource,
dispatcherManager: DispatcherManager,
totpCodeManager: TotpCodeManager,
fileManager: FileManager,
): AuthenticatorRepository = AuthenticatorRepositoryImpl(
authenticatorDiskSource,
totpCodeManager,
fileManager,
dispatcherManager,
)

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
package com.bitwarden.authenticator.data.authenticator.repository.model

/**
* Represents the result of a data export operation.
*/
sealed class ExportDataResult {

data object Success : ExportDataResult()

data object Error : ExportDataResult()

}
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import com.bitwarden.authenticator.ui.authenticator.feature.navbar.AUTHENTICATOR
import com.bitwarden.authenticator.ui.authenticator.feature.navbar.authenticatorNavBarDestination
import com.bitwarden.authenticator.ui.authenticator.feature.qrcodescan.navigateToQrCodeScanScreen
import com.bitwarden.authenticator.ui.authenticator.feature.search.navigateToSearch
import com.bitwarden.authenticator.ui.platform.feature.settings.export.navigateToExport
import com.bitwarden.authenticator.ui.platform.feature.tutorial.navigateToTutorial

const val AUTHENTICATOR_GRAPH_ROUTE = "authenticator_graph"
Expand Down Expand Up @@ -38,7 +39,7 @@ fun NavGraphBuilder.authenticatorGraph(
onNavigateToManualKeyEntry = { navController.navigateToManualCodeEntryScreen() },
onNavigateToEditItem = { navController.navigateToEditItem(itemId = it) },
onNavigateToTutorial = { navController.navigateToTutorial() },
onNavigateToExport = { /* TODO */ },
onNavigateToExport = { navController.navigateToExport() },
)
itemListingGraph(
navController = navController,
Expand All @@ -55,7 +56,7 @@ fun NavGraphBuilder.authenticatorGraph(
navController.navigateToEditItem(itemId = it)
},
navigateToTutorial = { navController.navigateToTutorial() },
navigateToExport = { /* TODO */ },
navigateToExport = { navController.navigateToExport() },
)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,9 @@ fun NavGraphBuilder.itemListingGraph(
onNavigateToSyncWithBitwardenScreen = {
/*navController.navigateToSyncWithBitwardenScreen()*/
},
onNavigateToImportScreen = { /*navController.navigateToImportScreen()*/ }
onNavigateToImportScreen = {
/*navController.navigateToImportScreen()*/
}
)
editItemDestination(
onNavigateBack = { navController.popBackStack() },
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -153,7 +153,7 @@ fun ItemListingScreen(
ItemListingAction.ConfirmDeleteClick(itemId = itemId),
)
}
}
},
)

BitwardenScaffold(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import androidx.navigation.NavGraphBuilder
import androidx.navigation.NavOptions
import androidx.navigation.navigation
import com.bitwarden.authenticator.ui.platform.base.util.composableWithRootPushTransitions
import com.bitwarden.authenticator.ui.platform.feature.settings.export.exportDestination

const val SETTINGS_GRAPH_ROUTE = "settings_graph"
private const val SETTINGS_ROUTE = "settings"
Expand All @@ -29,6 +30,9 @@ fun NavGraphBuilder.settingsGraph(
onNavigateToExport = onNavigateToExport,
)
}
exportDestination(
onNavigateBack = { navController.popBackStack() }
)
}
}

Expand Down
Loading

0 comments on commit 69da880

Please sign in to comment.