میتوانید با انجام ممیزی دسترسی به داده، اطلاعاتی درباره نحوه دسترسی برنامه و وابستگیهای آن به دادههای خصوصی کاربران به دست آورید. این فرآیند، در دستگاههایی که Android 11 (سطح API 30) و بالاتر را اجرا میکنند، در دسترس است، به شما امکان میدهد دسترسی به دادههای بالقوه غیرمنتظره را بهتر شناسایی کنید. برنامه شما میتواند نمونهای از AppOpsManager.OnOpNotedCallback
را ثبت کند، که میتواند هر بار که یکی از رویدادهای زیر رخ میدهد، اقداماتی را انجام دهد:
- کد برنامه شما به داده های خصوصی دسترسی دارد. برای کمک به شما در تعیین اینکه کدام بخش منطقی از برنامه شما رویداد را فراخوانی کرده است، میتوانید دسترسی به دادهها را با برچسب انتساب بررسی کنید .
- کد موجود در یک کتابخانه وابسته یا SDK به داده های خصوصی دسترسی دارد.
ممیزی دسترسی به داده در رشته ای که درخواست داده انجام می شود فراخوانی می شود. این بدان معنی است که اگر یک SDK یا کتابخانه شخص ثالث در برنامه شما یک API را فراخوانی کند که به داده های خصوصی دسترسی دارد، حسابرسی دسترسی به داده OnOpNotedCallback
شما اجازه می دهد اطلاعات مربوط به تماس را بررسی کند. معمولاً، این شیء برگشت به تماس با مشاهده وضعیت فعلی برنامه، مانند ردیابی پشته رشته فعلی، میتواند تشخیص دهد که تماس از برنامه شما بوده است یا SDK.
ورود به سیستم دسترسی به داده ها
برای انجام ممیزی دسترسی به دادهها با استفاده از نمونهای از AppOpsManager.OnOpNotedCallback
، منطق بازگشت به تماس را در مؤلفهای که قصد بررسی دسترسی دادهها را دارید، مانند متد onCreate()
یک فعالیت یا متد onCreate()
یک برنامه کاربردی پیادهسازی کنید.
قطعه کد زیر یک AppOpsManager.OnOpNotedCallback
را برای بررسی دسترسی به داده ها در یک فعالیت مشخص می کند:
کاتلین
override fun onCreate(savedInstanceState: Bundle?) { val appOpsCallback = object : AppOpsManager.OnOpNotedCallback() { private fun logPrivateDataAccess(opCode: String, trace: String) { Log.i(MY_APP_TAG, "Private data accessed. " + "Operation: $opCode\nStack Trace:\n$trace") } override fun onNoted(syncNotedAppOp: SyncNotedAppOp) { logPrivateDataAccess( syncNotedAppOp.op, Throwable().stackTrace.toString()) } override fun onSelfNoted(syncNotedAppOp: SyncNotedAppOp) { logPrivateDataAccess( syncNotedAppOp.op, Throwable().stackTrace.toString()) } override fun onAsyncNoted(asyncNotedAppOp: AsyncNotedAppOp) { logPrivateDataAccess(asyncNotedAppOp.op, asyncNotedAppOp.message) } } val appOpsManager = getSystemService(AppOpsManager::class.java) as AppOpsManager appOpsManager.setOnOpNotedCallback(mainExecutor, appOpsCallback) }
جاوا
@Override public void onCreate(@Nullable Bundle savedInstanceState, @Nullable PersistableBundle persistentState) { AppOpsManager.OnOpNotedCallback appOpsCallback = new AppOpsManager.OnOpNotedCallback() { private void logPrivateDataAccess(String opCode, String trace) { Log.i(MY_APP_TAG, "Private data accessed. " + "Operation: $opCode\nStack Trace:\n$trace"); } @Override public void onNoted(@NonNull SyncNotedAppOp syncNotedAppOp) { logPrivateDataAccess(syncNotedAppOp.getOp(), Arrays.toString(new Throwable().getStackTrace())); } @Override public void onSelfNoted(@NonNull SyncNotedAppOp syncNotedAppOp) { logPrivateDataAccess(syncNotedAppOp.getOp(), Arrays.toString(new Throwable().getStackTrace())); } @Override public void onAsyncNoted(@NonNull AsyncNotedAppOp asyncNotedAppOp) { logPrivateDataAccess(asyncNotedAppOp.getOp(), asyncNotedAppOp.getMessage()); } }; AppOpsManager appOpsManager = getSystemService(AppOpsManager.class); if (appOpsManager != null) { appOpsManager.setOnOpNotedCallback(getMainExecutor(), appOpsCallback); } }
متدهای onAsyncNoted()
و onSelfNoted()
در شرایط خاص فراخوانی می شوند:
در صورتی که دسترسی به داده در طول تماس API برنامه شما انجام نشود،
onAsyncNoted()
فراخوانی می شود. رایج ترین مثال زمانی است که برنامه شما یک شنونده را ثبت می کند و دسترسی به داده ها هر بار که تماس پاسخ شنونده فراخوانی می شود اتفاق می افتد.آرگومان
AsyncNotedOp
که بهonAsyncNoted()
منتقل می شود حاوی متدی به نامgetMessage()
است. این روش اطلاعات بیشتری در مورد دسترسی به داده ها ارائه می دهد. در مورد تماس های مکانی، پیام حاوی سیستم هویت-هش شنونده است.onSelfNoted()
در موارد بسیار نادری فراخوانی می شود که یک برنامه UID خود را بهnoteOp()
ارسال می کند.
دسترسی به داده های حسابرسی با برچسب انتساب
برنامه شما ممکن است چندین مورد استفاده اصلی داشته باشد، مانند اینکه به کاربران اجازه می دهد عکس بگیرند و این عکس ها را با مخاطبین خود به اشتراک بگذارند. اگر یک برنامه چند منظوره توسعه میدهید، میتوانید هنگام بررسی دسترسی به دادههای آن، برای هر قسمت از برنامه خود یک برچسب انتساب اعمال کنید. متن attributionTag
در اشیاء ارسال شده به فراخوانی های onNoted()
برمی گردد. این به شما کمک می کند تا دسترسی به داده ها را به راحتی به بخش های منطقی کد خود ردیابی کنید.
برای تعریف تگ های انتساب در برنامه خود، مراحل زیر را تکمیل کنید.
برچسبهای انتساب را در مانیفست اعلام کنید
اگر برنامه شما Android 12 (سطح API 31) یا بالاتر را هدف قرار می دهد، باید برچسب های انتساب را در فایل مانیفست برنامه خود با استفاده از قالب نشان داده شده در قطعه کد زیر اعلام کنید. اگر سعی می کنید از یک برچسب انتساب استفاده کنید که در فایل مانیفست برنامه خود اعلام نمی کنید، سیستم یک تگ null
برای شما ایجاد می کند و یک پیام را در Logcat ثبت می کند.
<manifest ...> <!-- The value of "android:tag" must be a literal string, and the value of "android:label" must be a resource. The value of "android:label" is user-readable. --> <attribution android:tag="sharePhotos" android:label="@string/share_photos_attribution_label" /> ... </manifest>
تگ های انتساب ایجاد کنید
در روش onCreate()
از فعالیتی که در آن به دادههایی دسترسی دارید، مانند فعالیتی که در آن مکان درخواست میکنید یا به فهرست مخاطبین کاربر دسترسی دارید، createAttributionContext()
را فراخوانی کنید، تگ attribution را که میخواهید با بخشی از خود مرتبط کنید، ارسال کنید. برنامه
قطعه کد زیر نحوه ایجاد یک تگ انتساب برای قسمت اشتراکگذاری مکان عکس از یک برنامه را نشان میدهد:
کاتلین
class SharePhotoLocationActivity : AppCompatActivity() { lateinit var attributionContext: Context override fun onCreate(savedInstanceState: Bundle?) { attributionContext = createAttributionContext("sharePhotos") } fun getLocation() { val locationManager = attributionContext.getSystemService( LocationManager::class.java) as LocationManager // Use "locationManager" to access device location information. } }
جاوا
public class SharePhotoLocationActivity extends AppCompatActivity { private Context attributionContext; @Override public void onCreate(@Nullable Bundle savedInstanceState, @Nullable PersistableBundle persistentState) { attributionContext = createAttributionContext("sharePhotos"); } public void getLocation() { LocationManager locationManager = attributionContext.getSystemService(LocationManager.class); if (locationManager != null) { // Use "locationManager" to access device location information. } } }
شامل برچسبهای انتساب در گزارشهای دسترسی
AppOpsManager.OnOpNotedCallback
خود را بهروزرسانی کنید تا گزارشهای برنامه شما شامل نام برچسبهای انتسابی باشد که تعریف کردهاید.
قطعه کد زیر منطق بهروزرسانی شدهای را نشان میدهد که تگهای انتساب را ثبت میکند:
کاتلین
val appOpsCallback = object : AppOpsManager.OnOpNotedCallback() { private fun logPrivateDataAccess( opCode: String, attributionTag: String, trace: String) { Log.i(MY_APP_TAG, "Private data accessed. " + "Operation: $opCode\n " + "Attribution Tag:$attributionTag\nStack Trace:\n$trace") } override fun onNoted(syncNotedAppOp: SyncNotedAppOp) { logPrivateDataAccess(syncNotedAppOp.op, syncNotedAppOp.attributionTag, Throwable().stackTrace.toString()) } override fun onSelfNoted(syncNotedAppOp: SyncNotedAppOp) { logPrivateDataAccess(syncNotedAppOp.op, syncNotedAppOp.attributionTag, Throwable().stackTrace.toString()) } override fun onAsyncNoted(asyncNotedAppOp: AsyncNotedAppOp) { logPrivateDataAccess(asyncNotedAppOp.op, asyncNotedAppOp.attributionTag, asyncNotedAppOp.message) } }
جاوا
@Override public void onCreate(@Nullable Bundle savedInstanceState, @Nullable PersistableBundle persistentState) { AppOpsManager.OnOpNotedCallback appOpsCallback = new AppOpsManager.OnOpNotedCallback() { private void logPrivateDataAccess(String opCode, String attributionTag, String trace) { Log.i("MY_APP_TAG", "Private data accessed. " + "Operation: $opCode\n " + "Attribution Tag:$attributionTag\nStack Trace:\n$trace"); } @Override public void onNoted(@NonNull SyncNotedAppOp syncNotedAppOp) { logPrivateDataAccess(syncNotedAppOp.getOp(), syncNotedAppOp.getAttributionTag(), Arrays.toString(new Throwable().getStackTrace())); } @Override public void onSelfNoted(@NonNull SyncNotedAppOp syncNotedAppOp) { logPrivateDataAccess(syncNotedAppOp.getOp(), syncNotedAppOp.getAttributionTag(), Arrays.toString(new Throwable().getStackTrace())); } @Override public void onAsyncNoted(@NonNull AsyncNotedAppOp asyncNotedAppOp) { logPrivateDataAccess(asyncNotedAppOp.getOp(), asyncNotedAppOp.getAttributionTag(), asyncNotedAppOp.getMessage()); } }; AppOpsManager appOpsManager = getSystemService(AppOpsManager.class); if (appOpsManager != null) { appOpsManager.setNotedAppOpsCollector(appOpsCollector); } }