مفاهیم کلیدی | محیط توسعه خود را تنظیم کنید | یک RE SDK بسازید | RE SDK را مصرف کنید | آزمایش و ساخت برای توزیع |
مفاهیم کلیدی | محیط توسعه خود را تنظیم کنید | یک RE SDK بسازید | RE SDK را مصرف کنید | آزمایش و ساخت برای توزیع |
یک SDK با قابلیت زمان اجرا بسازید
برای ساختن یک SDK با قابلیت زمان اجرا باید مراحل زیر را انجام دهید:
- ساختار پروژه خود را تنظیم کنید
- وابستگی های پروژه و ماژول خود را آماده کنید
- منطق کسب و کار SDK خود را اضافه کنید
- API های SDK را تعریف کنید
- یک نقطه ورودی برای SDK خود مشخص کنید
ساختار پروژه خود را تنظیم کنید
ما توصیه می کنیم که پروژه شما در ماژول های زیر سازماندهی شود:
- ماژول برنامه - برنامه آزمایشی که از آن برای آزمایش و توسعه SDK خود استفاده میکنید و نشاندهنده آن چیزی است که مشتریان برنامه واقعی شما دارند. برنامه شما باید به ماژول کتابخانه تبلیغات موجود ( SDK آگاه از زمان اجرا ) وابستگی داشته باشد.
- ماژول کتابخانه تبلیغات موجود (SDK آگاه از زمان اجرا) - یک ماژول کتابخانه Android که حاوی منطق SDK «غیر فعال در زمان اجرا» شما است، یک SDK به صورت ایستا پیوند خورده است.
- برای شروع، می توان توانایی ها را تقسیم کرد. به عنوان مثال، برخی از کدها را می توان توسط SDK موجود شما مدیریت کرد، و برخی را می توان به SDK فعال در زمان اجرا هدایت کرد.
- ماژول کتابخانه تبلیغاتی با قابلیت زمان اجرا - شامل منطق تجاری SDK فعال با زمان اجرا شما است. این را می توان در Android Studio به عنوان یک ماژول کتابخانه اندروید ایجاد کرد.
- ماژول ASB فعال در زمان اجرا - داده های بسته را برای بسته بندی کد SDK فعال شده با زمان اجرا در یک ASB تعریف می کند.
- باید به صورت دستی با استفاده از نوع com.android.privacy-sandbox-sdk ایجاد شود. می توانید این کار را با ایجاد یک دایرکتوری جدید انجام دهید.
- این ماژول نباید حاوی هیچ کد و فقط یک فایل خالی build.gradle با وابستگی به ماژول کتابخانه تبلیغاتی فعال در زمان اجرا شما باشد. محتوای این فایل در Prepare your SDK تعریف شده است.
- به یاد داشته باشید که این ماژول را در فایل settings.gradle و در ماژول کتابخانه تبلیغات موجود قرار دهید.
ساختار پروژه در این راهنما یک پیشنهاد است، می توانید ساختار متفاوتی را برای SDK خود انتخاب کنید و همان اصول فنی را اعمال کنید. همیشه میتوانید ماژولهای اضافی دیگری برای مدولار کردن کد در برنامه و ماژولهای کتابخانه ایجاد کنید.
SDK خود را آماده کنید
برای آماده سازی پروژه خود برای توسعه SDK با قابلیت زمان اجرا، ابتدا باید چند وابستگی ابزار و کتابخانه را تعریف کنید:
- کتابخانههای سازگاری SDK Runtime Backwards، که از دستگاههایی پشتیبانی میکنند که دارای جعبه ایمنی حریم خصوصی (اندروید ۱۳ و پایینتر) نیستند (
androidx.privacysandbox.sdkruntime:
) - کتابخانه های رابط کاربری برای پشتیبانی از ارائه تبلیغات (
androidx.privacysandbox.ui:
) - ابزارهای توسعه دهنده SDK برای پشتیبانی از اعلان API SDK و تولید شیم (
androidx.privacysandbox.tools:
)
این پرچم را به فایل gradle.properties پروژه خود اضافه کنید تا قابلیت ایجاد SDK با قابلیت زمان اجرا فعال شود.
# This enables the Privacy Sandbox for your project on Android Studio. android.experimental.privacysandboxsdk.enable=true android.experimental.privacysandboxsdk.requireServices=false
build.gradle پروژه خود را تغییر دهید تا کتابخانه های Jetpack کمکی و وابستگی های دیگر را شامل شود:
// Top-level build file where you can add configuration options common to all sub-projects/modules. buildscript { ext.kotlin_version = '1.9.10' ext.ksp_version = "$kotlin_version-1.0.13" ext.privacy_sandbox_activity_version = "1.0.0-alpha01" ext.privacy_sandbox_sdk_runtime_version = "1.0.0-alpha13" ext.privacy_sandbox_tools_version = "1.0.0-alpha09" ext.privacy_sandbox_ui_version = "1.0.0-alpha09" repositories { mavenCentral() } dependencies { classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" } } plugins { id 'com.android.application' version '8.4.0-alpha13' apply false id 'com.android.library' version '8.4.0-alpha13' apply false // These two plugins do annotation processing and code generation for the sdk-implementation. id 'androidx.privacysandbox.library' version '1.0.0-alpha02' apply false id 'com.google.devtools.ksp' version "$ksp_version" apply false id 'org.jetbrains.kotlin.jvm' version '1.9.10' apply false } task clean(type: Delete) { delete rootProject.buildDir }
فایل build.gradle را در ماژول کتابخانه تبلیغاتی با قابلیت اجرا (RE SDK) بهروزرسانی کنید تا این وابستگیها را شامل شود.
dependencies { // This allows Android Studio to parse and validate your SDK APIs. ksp "androidx.privacysandbox.tools:tools-apicompiler:$privacy_sandbox_tools_version" // This contains the annotation classes to decorate your SDK APIs. implementation "androidx.privacysandbox.tools:tools:$privacy_sandbox_tools_version" // This is runtime dependency required by the generated server shim code for // backward compatibility. implementation "androidx.privacysandbox.sdkruntime:sdkruntime-provider:$privacy_sandbox_sdk_runtime_version" // These are runtime dependencies required by the generated server shim code as // they use Kotlin. implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:1.7.1" implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.7.1' // This is the core part of the UI library to help with UI notifications. implementation "androidx.privacysandbox.ui:ui-core:$privacy_sandbox_ui_version" // This helps the SDK open sessions for the ad. implementation "androidx.privacysandbox.ui:ui-provider:$privacy_sandbox_ui_version" // This is needed if your SDK implements mediation use cases implementation "androidx.privacysandbox.ui:ui-client:$privacy_sandbox_ui_version" }
فایل build.gradle را در ماژول ASB فعال در زمان اجرا با موارد زیر جایگزین کنید:
plugins { id 'com.android.privacy-sandbox-sdk' } android { compileSdk 34 minSdk 21 bundle { // This is the package name of the SDK that you want to publish. // This is used as the public identifier of your SDK. // You use this later on to load the runtime-enabled SDK packageName = '<package name of your runtime-enabled SDK>' // This is the version of the SDK that you want to publish. // This is used as the public identifier of your SDK version. setVersion(1, 0, 0) // SDK provider defined in the SDK Runtime library. // This is an important part of the future backwards compatibility // support, most SDKs won't need to change it. sdkProviderClassName = "androidx.privacysandbox.sdkruntime.provider.SandboxedSdkProviderAdapter" // This is the class path of your implementation of the SandboxedSdkProviderCompat class. // It's the implementation of your runtime-enabled SDK's entry-point. // If you miss this step, your runtime-enabled SDK will fail to load at runtime: compatSdkProviderClassName = "<your-sandboxed-sdk-provider-compat-fully-qualified-class-name>" } } dependencies { // This declares the dependency on your runtime-enabled ad library module. include project(':<your-runtime-enabled-ad-library-here>') }
فایل build.gradle را در ماژول کتابخانه تبلیغات موجود خود (RA SDK) بهروزرسانی کنید تا وابستگیهای زیر را شامل شود:
dependencies { // This declares the client's dependency on the runtime-enabled ASB module. // ⚠️ Important: We depend on the ASB module, not the runtime-enabled module. implementation project(':<your-runtime-enabled-asb-module-here>') // Required for backwards compatibility on devices where SDK Runtime is unavailable. implementation "androidx.privacysandbox.sdkruntime:sdkruntime-client:$privacy_sandbox_sdk_runtime_version" // This is required to display banner ads using the SandboxedUiAdapter interface. implementation "androidx.privacysandbox.ui:ui-core:$privacy_sandbox_ui_version" implementation "androidx.privacysandbox.ui:ui-client:$privacy_sandbox_ui_version" // This is required to use SDK ActivityLaunchers. implementation "androidx.privacysandbox.activity:activity-core:$privacy_sandbox_activity_version" implementation "androidx.privacysandbox.activity:activity-client:$privacy_sandbox_activity_version" }
منطق کسب و کار SDK را اضافه کنید
منطق کسب و کار SDK خود را همانطور که به طور منظم در ماژول کتابخانه تبلیغاتی فعال با زمان اجرا انجام می دهید، پیاده سازی کنید.
اگر یک SDK موجود دارید که در حال انتقال آن هستید، در این مرحله به اندازه منطق کسب و کار، رابط، و عملکردهای رو به رو سیستم خود را جابه جا کنید، اما در آینده انتقال کامل را در نظر بگیرید.
اگر نیاز به دسترسی به فضای ذخیرهسازی، شناسه تبلیغات Google Play، یا شناسه مجموعه برنامه دارید، بخشهای زیر را بخوانید:
از API های ذخیره سازی در SDK خود استفاده کنید
SDK های موجود در SDK Runtime دیگر نمی توانند به حافظه داخلی برنامه دسترسی داشته باشند، بخوانند یا بنویسند و برعکس.
SDK Runtime فضای ذخیره سازی داخلی خود را جدا از برنامه اختصاص داده است.
SDKها میتوانند با استفاده از APIهای ذخیرهسازی فایل روی شی Context
که توسط SandboxedSdkProvider#getContext()
بازگردانده شده است، به این حافظه داخلی جداگانه دسترسی پیدا کنند.
SDK ها فقط می توانند از حافظه داخلی استفاده کنند، بنابراین فقط API های ذخیره سازی داخلی، مانند Context.getFilesDir()
یا Context.getCacheDir()
کار می کنند. نمونههای بیشتری را در دسترسی از حافظه داخلی مشاهده کنید.
دسترسی به حافظه خارجی از SDK Runtime پشتیبانی نمی شود. فراخوانی APIها برای دسترسی به حافظه خارجی یا یک استثنا ایجاد می کند یا null را برمی گرداند. لیست زیر شامل چند نمونه است:
- دسترسی به فایل ها با استفاده از Storage Access Framework یک SecurityException ایجاد می کند.
-
getExternalFilsDir()
همیشه null را برمیگرداند.
شما باید از Context
برگردانده شده توسط SandboxedSdkProvider.getContext()
برای ذخیره سازی استفاده کنید. استفاده از API ذخیرهسازی فایل در هر نمونه دیگری از شی Context
، مانند زمینه برنامه، تضمین نمیشود که در همه موقعیتها همانطور که انتظار میرود کار کند.
قطعه کد زیر نحوه استفاده از فضای ذخیره سازی در SDK Runtime را نشان می دهد:
class SdkServiceImpl(private val context: Context) : SdkService { override suspend fun getMessage(): String = "Hello from Privacy Sandbox!" override suspend fun createFile(sizeInMb: Int): String { val path = Paths.get( context.dataDir.path, "file.txt" ) withContext(Dispatchers.IO) { Files.deleteIfExists(path) Files.createFile(path) val buffer = ByteArray(sizeInMb * 1024 * 1024) Files.write(path, buffer) } val file = File(path.toString()) val actualFileSize: Long = file.length() / (1024 * 1024) return "Created $actualFileSize MB file successfully" } }
در حافظه داخلی جداگانه برای هر زمان اجرا SDK، هر SDK فهرست ذخیره سازی مخصوص به خود را دارد. فضای ذخیره سازی برای هر SDK یک تفکیک منطقی از حافظه داخلی SDK Runtime است که به محاسبه مقدار فضای ذخیره سازی هر SDK کمک می کند.
همه APIهای ذخیره سازی داخلی در شی Context
یک مسیر ذخیره سازی برای هر SDK برمی گردانند.
به شناسه تبلیغاتی ارائه شده توسط خدمات Google Play دسترسی داشته باشید
اگر SDK شما نیاز به دسترسی به شناسه تبلیغاتی ارائه شده توسط خدمات Google Play دارد، از AdIdManager#getAdId()
برای بازیابی مقدار به صورت ناهمزمان استفاده کنید.
به شناسه مجموعه برنامه ارائه شده توسط خدمات Google Play دسترسی داشته باشید
اگر SDK شما نیاز به دسترسی به شناسه مجموعه برنامه ارائه شده توسط خدمات Google Play دارد، از AppSetIdManager#getAppSetId()
برای بازیابی مقدار به صورت ناهمزمان استفاده کنید.
API های SDK را اعلام کنید
برای اینکه SDK فعال در زمان اجرا شما خارج از زمان اجرا قابل دسترسی باشد، باید API هایی را تعریف کنید که کلاینت ها (RA SDK یا برنامه مشتری) می توانند مصرف کنند.
از حاشیه نویسی برای اعلام این رابط ها استفاده کنید.
حاشیه نویسی ها
API های SDK باید در Kotlin به عنوان رابط ها و کلاس های داده با استفاده از حاشیه نویسی های زیر اعلان شوند:
حاشیه نویسی ها | |
---|---|
@PrivacySandboxService |
|
@PrivacySandboxInterface |
|
@PrivacySandboxValue |
|
@PrivacySandboxCallback |
|
شما باید این رابطها و کلاسها را در هر جایی از ماژول کتابخانه تبلیغاتی فعالشده با زمان اجرا تعریف کنید.
استفاده از این حاشیه نویسی را در بخش های بعدی مشاهده کنید.
@PrivacySandboxService
@PrivacySandboxService interface SdkService { suspend fun getMessage(): String suspend fun createFile(sizeInMb: Int): String suspend fun getBanner(request: SdkBannerRequest, requestMediatedAd: Boolean): SdkSandboxedUiAdapter? suspend fun getFullscreenAd(): FullscreenAd }
@PrivacySandboxInterface
@PrivacySandboxInterface interface SdkSandboxedUiAdapter : SandboxedUiAdapter
@PrivacySandboxValue
@PrivacySandboxValue data class SdkBannerRequest( /** The package name of the app. */ val appPackageName: String, /** * An [SdkActivityLauncher] used to launch an activity when the banner is clicked. */ val activityLauncher: SdkActivityLauncher, /** * Denotes if a WebView banner ad needs to be loaded. */ val isWebViewBannerAd: Boolean )
@PrivacySandboxCallback
@PrivacySandboxCallback interface InAppMediateeSdkInterface { suspend fun show() }
انواع پشتیبانی شده
API های SDK فعال شده در زمان اجرا از انواع زیر پشتیبانی می کنند:
- همه انواع اولیه در زبان برنامه نویسی جاوا (مانند int، long، char، boolean و غیره)
- رشته
- رابط های Kotlin حاشیه نویسی شده با
@PrivacySandboxInterface
یا@PrivacySandboxCallback
- کلاس های داده Kotlin با
@PrivacySandboxValue
حاشیه نویسی شده اند - java.lang.List - همه عناصر موجود در لیست باید یکی از انواع داده های پشتیبانی شده باشند
چند اخطار اضافی وجود دارد:
- کلاس های داده حاشیه نویسی شده با
@PrivacySandboxValue
نمی توانند دارای فیلدهایی از نوع@PrivacySandboxCallback
باشند - انواع برگشتی نمی توانند حاوی انواع حاشیه نویسی شده با
@PrivacySandboxCallback
باشند - فهرست نمی تواند حاوی عناصری از انواع حاشیه نویسی شده با
@PrivacySandboxInterface
یا@PrivacySandboxCallback
باشد
API های ناهمزمان
از آنجایی که API های SDK همیشه با یک فرآیند جداگانه تماس برقرار می کنند، باید اطمینان حاصل کنیم که این تماس ها رشته تماس مشتری را مسدود نمی کنند.
برای دستیابی به این هدف، همه روشها در رابطهای حاشیهنویسی شده با @PrivacySandboxService
، @PrivacySandboxInterface
و @PrivacySandboxCallback
باید بهصراحت بهعنوان APIهای ناهمزمان اعلام شوند.
API های ناهمزمان را می توان به دو روش در Kotlin پیاده سازی کرد:
- از توابع تعلیق استفاده کنید.
- تماسهایی را بپذیرید که پس از اتمام عملیات یا سایر رویدادها در حین پیشرفت عملیات مطلع میشوند. نوع برگشتی تابع باید یک واحد باشد.
استثنائات
APIهای SDK هیچ شکلی از استثناهای بررسی شده را پشتیبانی نمی کنند.
کد شیم تولید شده هرگونه استثنا در زمان اجرا را که توسط SDK پرتاب می شود را می گیرد و آنها را به عنوان PrivacySandboxException
با اطلاعاتی در مورد علت در داخل آن به مشتری می فرستد.
کتابخانه UI
اگر واسطهایی دارید که تبلیغات را نشان میدهند، مانند یک بنر، باید رابط SandboxedUiAdapter
را نیز برای فعال کردن جلسات باز کردن آگهی بارگذاری شده پیادهسازی کنید.
این جلسات یک کانال جانبی بین مشتری و SDK تشکیل می دهند و دو هدف اصلی را برآورده می کنند:
- هر زمان که تغییر رابط کاربری رخ می دهد، اعلان ها را دریافت کنید.
- مشتری را از هرگونه تغییر در ارائه رابط کاربری مطلع کنید.
از آنجایی که مشتری می تواند از رابط مشروح شده با @PrivacySandboxService
برای ارتباط با SDK شما استفاده کند، هر API برای بارگیری تبلیغات می تواند به این رابط اضافه شود.
هنگامی که مشتری برای بارگیری تبلیغ درخواست می کند، آگهی را بارگیری کرده و نمونه ای از رابط پیاده سازی SandboxedUiAdapter
را برگردانید. این به مشتری اجازه می دهد تا جلسات بازگشایی آن تبلیغ را درخواست کند.
وقتی مشتری برای باز کردن جلسه درخواست میکند، SDK فعال با زمان اجرا شما میتواند با استفاده از پاسخ آگهی و زمینه ارائه شده، یک نمای تبلیغاتی ایجاد کند.
برای رسیدن به این هدف، کلاسی ایجاد کنید که رابط SandboxedUiAdapter.Session
را پیاده سازی کند و هنگامی که SandboxedUiAdapter.openSession()
فراخوانی می شود، مطمئن شوید که client.onSessionOpened()
را فراخوانی می کنید و نمونه ای از کلاس Session
را به عنوان پارامتر ارسال می کنید.
class SdkSandboxedUiAdapterImpl(
private val sdkContext: Context,
private val request: SdkBannerRequest,
) : SdkSandboxedUiAdapter {
override fun openSession(
context: Context,
windowInputToken: IBinder,
initialWidth: Int,
initialHeight: Int,
isZOrderOnTop: Boolean,
clientExecutor: Executor,
client: SandboxedUiAdapter.SessionClient
) {
val session = SdkUiSession(clientExecutor, sdkContext, request)
clientExecutor.execute {
client.onSessionOpened(session)
}
}
}
این کلاس همچنین هر زمان که تغییر رابط کاربری ایجاد شود اعلانهایی دریافت میکند. میتوانید از این کلاس برای تغییر اندازه آگهی استفاده کنید، یا بدانید که چه زمانی پیکربندی تغییر کرده است.
درباره APIهای ارائه رابط کاربری در زمان اجرا بیشتر بیاموزید.
پشتیبانی از فعالیت
برای شروع فعالیتهای متعلق به SDK از جعبه ایمنی حریم خصوصی، باید API SDK را تغییر دهید تا یک شی SdkActivityLauncher
را که همچنین توسط کتابخانه UI ارائه شده است، دریافت کنید.
به عنوان مثال، API SDK زیر باید فعالیتها را راهاندازی کند، بنابراین انتظار پارامتر SdkActivityLauncher
را دارد:
@PrivacySandboxInterface
interface FullscreenAd {
suspend fun show(activityLauncher: SdkActivityLauncher)
}
نقطه ورود SDK
کلاس انتزاعی SandboxedSdkProvider
API را کپسوله می کند که SDK Runtime از آن برای تعامل با SDK های بارگذاری شده در آن استفاده می کند.
یک SDK فعال با زمان اجرا باید این کلاس انتزاعی را پیاده سازی کند تا یک نقطه ورودی برای زمان اجرا SDK ایجاد کند تا بتواند با آن ارتباط برقرار کند.
برای پشتیبانی از سازگاری با عقب، ما کلاس های زیر را معرفی کرده ایم:
-
SandboxedSdkProviderAdapter
، کهSandboxedSdkProvider
را گسترش میدهد و درخواستهای بارگیری SDK را بدون توجه به در دسترس بودن SDK Runtime رسیدگی میکند. این به صورت داخلی استفاده می شود و در ماژول ASB اعلام شده است. -
SandboxedSdkProviderCompat
، یک کلاس انتزاعی که از رابطSandboxedSdkProvider
تقلید می کند.
درباره سازگاری به عقب برای زمان اجرا SDK بیشتر بیاموزید.
ابزارهای تولید شیم لایه دیگری از انتزاع اضافه می کنند: آنها یک کلاس انتزاعی به نام AbstractSandboxedSdkProvider
را با استفاده از رابطی که با @PrivacySandboxService
حاشیه نویسی کردید ایجاد می کنند.
این کلاس SandboxedSdkProviderCompat
را گسترش می دهد و تحت همان بسته رابط حاشیه نویسی شما قرار دارد.
// Auto-generated code.
abstract class AbstractSandboxedSdkProvider : SandboxedSdkProviderCompat {
abstract fun createMySdk(context: Context): MySdk
}
این کلاس تولید شده یک روش کارخانه انتزاعی واحد را نشان می دهد که یک Context
را می گیرد و انتظار دارد رابط مشروح نقطه ورودی شما برگردانده شود.
نام این روش از رابط کاربری @PrivacySandboxService
شما گرفته create
است. به عنوان مثال، اگر رابط شما MySdk
نام دارد، ابزارها createMySdk
ایجاد می کنند.
برای اتصال کامل نقطه ورودی خود، باید یک پیاده سازی از واسط حاشیه نویسی @PrivacySandboxService
خود را در SDK فعال در زمان اجرا به AbstractSandboxedSdkProvider
ایجاد شده ارائه دهید.
class MySdkSandboxedSdkProvider : AbstractSandboxedSdkProvider() {
override fun createMySdk(context: Context): MySdk = MySdkImpl(context)
}
تغییرات در ماژول ASB
شما باید نام کلاس کاملاً واجد شرایط پیاده سازی SandboxedSdkProviderCompat
را در فیلد compatSdkProviderClassName
build.gradle ماژول ASB خود اعلام کنید.
این کلاسی است که در مرحله قبل پیاده سازی کردید و build.gradle را در ماژول ASB خود به صورت زیر تغییر می دهید:
bundle {
packageName = '<package name of your runtime-enabled SDK>'
setVersion(1, 0, 0)
// SDK provider defined in the SDK Runtime library.
sdkProviderClassName = "androidx.privacysandbox.sdkruntime.provider.SandboxedSdkProviderAdapter"
// This is the class that extends AbstractSandboxedSdkProvider,
// MySdkSandboxProvider as per the example provided.
compatSdkProviderClassName = "com.example.mysdk.MySdkSandboxProvider"
}
مرحله 2 : محیط توسعه خود را تنظیم کنیدمرحله 4 : از SDK فعال با زمان اجرا استفاده کنید