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 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
|
// Copyright 2025 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef BASE_ANDROID_VIRTUAL_DOCUMENT_PATH_H_
#define BASE_ANDROID_VIRTUAL_DOCUMENT_PATH_H_
#include <jni.h>
#include <optional>
#include <string>
#include "base/android/scoped_java_ref.h"
#include "base/containers/span.h"
namespace base::files_internal {
// Represents and operates on a virtual path for Android's Storage Access
// Framework (SAF).
//
// `base::FilePath` can store path-like strings, including `content://` URIs.
// However, applying string manipulations (like `Append`) to a `FilePath`
// that holds a `content://` URI often results in an invalid URI, as these
// URIs are not simple hierarchical paths.
//
// To address this, the `/SAF/...` virtual path format was introduced. This
// format is specifically designed to be safely manipulated by `FilePath`'s
// string operations. The path can represent both file and directory paths.
//
// This class, `VirtualDocumentPath`, is an object representation of a
// complete and immutable virtual document path. It is created by parsing a
// `/SAF/...` string. The class itself does not support path manipulation; its
// role is to interpret the virtual document path and execute operations against
// it, such as resolving it to a content URI (`ResolveToContentUri`) or
// performing file I/O (`WriteFile`).
//
// The virtual path format it parses is:
// /SAF/<authority>/tree/<documentID>/<relativePath>
//
// USAGE
//
// This class is primarily intended for internal use within the `//base/files`
// file API implementation.
//
// Code outside of `//base/files` should remain unaware of
// `VirtualDocumentPath`. Path construction should be done using
// `base::FilePath`. The resulting `FilePath` can then be passed to
// `//base/files` helper functions which, internally, may use
// `VirtualDocumentPath::Parse()` to interpret the path and perform an
// operation.
//
// EXAMPLE (for //base/files developers)
//
// To operate on a SAF path, first construct the full path using
// `base::FilePath`, then parse it into a `VirtualDocumentPath` object.
//
// Convert the `FilePath` storing a document tree URI to a `FilePath` storing
// a virtual document path:
//
// base::FilePath dir(
// "content://com.android.externalstorage.documents/tree/primary:A%2FB");
// base::FilePath dir_vp = *dir.ResolveToVirtualDocumentPath();
//
// Construct the full path string using FilePath:
//
// base::FilePath file_vp = dir_vp.Append("c.txt");
//
// Parse the virtual document path string into a VirtualDocumentPath object:
//
// VirtualDocumentPath file_vpath = *VirtualDocumentPath::Parse(
// file_vp.value());
//
// Use the object to perform an operation:
//
// file_vpath->WriteFile(some_data);
//
// To perform I/O via other Android APIs, the virtual path can be resolved to
// a `content://` URI using `ResolveToContentUri()`:
//
// base::FilePath file(*file_vpath.ResolveToContentUri());
class VirtualDocumentPath {
public:
VirtualDocumentPath(const VirtualDocumentPath& path);
VirtualDocumentPath& operator=(const VirtualDocumentPath& path);
~VirtualDocumentPath();
// Parses virtual path "/SAF/..." to `VirtualDocumentPath` or resolves a tree
// URI (a content URI that represents a document tree) into
// `VirtualDocumentPath`.
// See
// https://developer.android.com/reference/android/provider/DocumentsContract
// for more about document tree URIs.
static std::optional<VirtualDocumentPath> Parse(const std::string& path);
// Resolves it to a content URI. If the file does not exist, it will return
// nullopt. If it returns a value, it will not be an empty string.
std::optional<std::string> ResolveToContentUri() const;
// Returns string representation of the instance. See the class level comment
// for details.
std::string ToString() const;
// Makes directory represented by the virtual path.
// It returns whether the directory has been successfully created. If the file
// already exists, it does nothing and returns false.
bool Mkdir(mode_t mode) const;
// Writes data to the file represented by the virtual path. If the file
// already exists its content is truncated first. It returns true if the data
// has been successfully written, and false otherwise.
bool WriteFile(span<const uint8_t> data) const;
// Creates an empty file if it does not exist and its parent directory exists.
// If the file exists or created, it returns a pair of two values where the
// first value is the content URI, and the second is a bool which is true if
// the file has been created and false if the file already existed.
std::optional<std::pair<std::string, bool>> CreateOrOpen() const;
private:
explicit VirtualDocumentPath(const base::android::JavaRef<jobject>& obj);
base::android::ScopedJavaGlobalRef<jobject> obj_;
};
} // namespace base::files_internal
#endif // BASE_ANDROID_VIRTUAL_DOCUMENT_PATH_H_
|