iOS 是如何获取夜间模式启动图缓存路径?

移动开发
本文通过对 applicationState.db 进行一系列的分析,最终实现了在模拟器下获取获取夜间模式启动图缓存路径的诉求。

你是否了解过iOS 是如何获取夜间模式启动图缓存路径??

百度APP技术团队曾经发布过一篇深夜暗坑 - iOS启动图异常修复方案。

该文章分享了一些关于启动图的研究,但是通过文章的内容,我们仍然无法解决这个问题:iOS 是如何获取夜间模式启动图缓存路径?

[[353923]]

经过一系列的研究,作者获了模拟器场景下的方案,现分享作者的研究记录。

方案一:尝试通过启动图文件名判断缓存图是否属于夜间模式

我们首先对 深夜暗坑 - iOS启动图异常修复方案 提供的信息进行提炼。

原文提供了以下2个信息:

  • 缓存启动图的文件名具有规则,但其规则我们不得而知
  • 4 张启动图的文件名
  1. ├── 1FFD332B-EBA0-40C9-8EEE-BEC9AEF7C41A@3x.ktx 
  2. ├── 96920D11-6312-4D69-BBDB-AFBB52DBDDB3@3x.ktx 
  3. ├── 98F7B5B1-5B3B-478B-93A8-ED3DE6492AD1@3x.ktx 
  4. └── D9D48845-8565-42CE-A834-479CC9CC8BAD@3x.ktx 

通过 4 个文件名,我们可以发现4张图片的命名都符合以下规则:

  1. xxxxxxxx-xxxx-4xxx-xxxx-xxxxxxxxxxxx 

再结合苹果官方文档 NSUUID 的内容:

  1. `NSUUID`conform to RFC 4122 version 4 and are created with random bytes. 

我们可以得到以下结论:

  • 4 个文件名的都是通过 NSUUID 动态生成
  • 文件名只包含版本 4,不再包含其它有效的信息

方案二:通过系统文件进行分析

方案一失败后,我们猜测 iOS 是通过其它方式保存夜间模式启动图缓存路径信息。

经过一系列的测试,作者发现了 applicationState.db 文件。

applicationState.db

系统会通过 applicationState.db 保存程序状态等各类信息,当然,也会包括夜间模式启动图缓存路径信息。

本文分析的文件位于 ~/Library/Developer/CoreSimulator/Devices/1F9B22C5-E446-4881-AFE4-3373E3513C59/data/Library/FrontBoard/applicationState.db

其中,1F9B22C5-E446-4881-AFE4-3373E3513C59代表 iOS 模拟器的设备ID。

模拟器的完整ID列表可以通过命令 plutil -p ~/Library/Developer/CoreSimulator/Devices/device_set.plist 查看

测试环境

为了方便对系统文件进行分析,本文以 iOS 14 模拟器为目标进行分析。

版本信息如下所示:

  1. (lldb) platform status 
  2.   Platform: ios-simulator 
  3.     Triple: x86_64h-apple-macosx 
  4. OS Version: 10.15.6 (19G2021) 
  5.     Kernel: Darwin Kernel Version 19.6.0: Thu Jun 18 20:49:00 PDT 2020; root:xnu-6153.141.1~1/RELEASE_X86_64 
  6.   Hostname: 127.0.0.1 
  7. WorkingDir: / 
  8.   SDK Path: "/Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator.sdk" 
  9. No devices are available. 

另外,后续的代码会假定 Bundle identifier 是 test.SplashTest

解析 applicationState.db

先尝试通过 file命令获取 applicationState.db文件类型:

  1. file ~/Library/Developer/CoreSimulator/Devices/1F9B22C5-E446-4881-AFE4-3373E3513C59/data/Library/FrontBoard/applicationState.db 

输出:

  1. SQLite 3.x databaselast written using SQLite version 3032003 

测试成功,通过 file 命令的输出可以看到文件类型是 SQLite 3.x。

db 结构

下面,再通过 SQLite 相关的工具对该文件进行dump,我们可以得到以下信息:

  1. sqlite> .schema 
  2. CREATE TABLE schema(version INT NOT NULL); 
  3. CREATE TABLE key_tab (id INTEGER PRIMARY KEYkey TEXT NOT NULLUNIQUE(key)); 
  4. CREATE TABLE application_identifier_tab (id INTEGER PRIMARY KEY, application_identifier TEXT NOT NULLUNIQUE(application_identifier)); 
  5. CREATE TABLE kvs (    id INTEGER PRIMARY KEY,    application_identifier INT REFERENCES application_identifier_tab(id),    key INT REFERENCES key_tab(id),    value BLOB,    UNIQUE(application_identifier, key)); 
  6. CREATE INDEX kvs_keys ON kvs(key); 
  7. CREATE INDEX kvs_application_identifiers ON kvs(application_identifier); 
  8. CREATE VIEW kvs_debug AS     SELECT application_identifier_tab.application_identifier, key_tab.key, value FROM application_identifier_tab, key_tab, kvs WHERE         kvs.application_identifier=application_identifier_tab.id         AND kvs.key=key_tab.id 
  9. /* kvs_debug(application_identifier,"key",value) */; 

application_identifier_tab 保存了设备安装的应用列表( id 是主键,application_identifier 是 APP 的Bundle identifier)

key_tab 负责记录常量字符串。

经过测试,夜间模式启动图缓存路径的路径属于 XBApplicationSnapshotManifest。

 

  1. sqlite> .schema key_tab 
  2. CREATE TABLE key_tab (id INTEGER PRIMARY KEYkey TEXT NOT NULLUNIQUE(key)); 
  3. sqlite> .width 2 50 
  4. sqlite> select * from key_tab; 
  5. id  key 
  6. --  -------------------------------------------------- 
  7. 1   SBLaunchImageIngestionInfo 
  8. 2   XBApplicationSnapshotManifest 
  9. 3   _SBScenes 
  10. 4   SBApplicationShortcutItems 
  11. 5   compatibilityInfo 
  12. 6   SBApplicationRecentlyUpdated 
  13. 7   SBApplicationRecentlyUpdatedTimerStartDate 

kvs_debug 负责串联上面的3个表:

  1. kvs.application_identifier=application_identifier_tab.id 
  2. kvs.key=key_tab.id 

通过 test.SplashTest,可以获取到4个结果,其中第2负责保存快照相关信息

  1. sqlite> .width 15 32 8 
  2. sqlite> SELECT * FROM kvs_debug WHERE application_identifier = 'test.SplashTest'
  3. application_ide  key                               value 
  4. ---------------  --------------------------------  -------- 
  5. test.SplashTest  _SBScenes                         bplist00 
  6. test.SplashTest  XBApplicationSnapshotManifest     bplist00 
  7. test.SplashTest  SBApplicationRecentlyUpdated      0 
  8. test.SplashTest  compatibilityInfo                 bplist00 

导出XBApplicationSnapshotManifest

经过一番研究,我们发现 XBApplicationSnapshotManifest 对应的value 就是 SplashBoard 库 XBApplicationSnapshotManifestImpl 类的持久化结果。

所以,我们可以通过通过以下代码,对 value 的内容进行 dump。

  1. +(void)load {     
  2.     void *lib  =  dlopen("/Applications/Xcode.app/Contents/Developer/Platforms/iPhoneOS.platform/Library/Developer/CoreSimulator/Profiles/Runtimes/iOS.simruntime/Contents/Resources/RuntimeRoot/System/Library/PrivateFrameworks/SplashBoard.framework/SplashBoard", RTLD_NOW); 
  3.     printf("%p", lib); 
  4.     [self dump:@"/Users/test/Library/Developer/CoreSimulator/Devices/1F9B22C5-E446-4881-AFE4-3373E3513C59/data/Library/FrontBoard/XBApplicationSnapshotManifest.plist"]; 
  5.  
  6. +(void)dump:(NSString *)path { 
  7.     NSData *data0 = [NSData dataWithContentsOfFile:path]; 
  8.     NSPropertyListFormat f = -1; 
  9.     NSError *error = nil; 
  10.     NSData *data1 = [NSPropertyListSerialization propertyListWithData:data0 options:NSPropertyListReadStreamError format:&f error:&error]; 
  11.     if (f==kCFPropertyListXMLFormat_v1_0) { 
  12.         NSLog(@"kCFPropertyListXMLFormat_v1_0"); 
  13.     } 
  14.     id obj = [NSKeyedUnarchiver unarchiveTopLevelObjectWithData:data1 error:&error]; 
  15.     NSLog(@"%@",obj); 

dump 结果:

  1. <XBApplicationSnapshotManifestImpl: 0x600000050d80; clientCount: 0> { 
  2.     containerIdentity = 0x0; 
  3.     snapshots = { 
  4.         <XBApplicationSnapshotGroup: 0x600002e65ea0; identifier: test.SplashTest - {DEFAULT GROUP}> { 
  5.             <XBApplicationSnapshot: 0x7fcb53f04d90; identifier: CE275D00-5732-4AFD-88FD-00BAE541EC12; launchInterfaceIdentifier: __from_UILaunchStoryboardName__; contentType: GeneratedDefault; referenceSize: {375, 812}; interfaceOrientation: LandscapeLeft; userInterfaceStyle: Dark> { 
  6.                 creationDate = September 27, 2020 at 6:08:50 PM GMT+8; 
  7.                 keepsImageAccessUntilExpiration = NO
  8.                 hasGenerationContext = NO
  9.                 context = { 
  10.                     contentType = GeneratedDefault; 
  11.                     fullScreen = YES; 
  12.                     referenceSize = {375, 812}; 
  13.                     interfaceOrientation = LandscapeLeft; 
  14.                     userInterfaceStyle = Dark; 
  15.                     additionalContext = { 
  16.                         statusBarSettings = <XBStatusBarSettings: 0x600002c580f0; hidden: YES; style: 0x0; backgroundActivityEnabled: NO>; 
  17.                     } 
  18.                 } 
  19.                 imageContext = { 
  20.                     scale = 3.0; 
  21.                     opaque = YES; 
  22.                     fileRelativeLocation = default
  23.                     fileFormat = png; 
  24.                 } 
  25.             }; 
  26.             <XBApplicationSnapshot: 0x7fcb57004830; identifier: B9DAB53E-29D9-47D2-873E-5772DE9220D1; launchInterfaceIdentifier: __from_UILaunchStoryboardName__; contentType: GeneratedDefault; referenceSize: {375, 812}; interfaceOrientation: Portrait; userInterfaceStyle: Light> { 
  27.                 creationDate = September 27, 2020 at 6:08:50 PM GMT+8; 
  28.                 lastUsedDate = September 27, 2020 at 6:08:50 PM GMT+8; 
  29.                 keepsImageAccessUntilExpiration = NO
  30.                 hasGenerationContext = NO
  31.                 context = { 
  32.                     contentType = GeneratedDefault; 
  33.                     fullScreen = YES; 
  34.                     referenceSize = {375, 812}; 
  35.                     interfaceOrientation = Portrait; 
  36.                     userInterfaceStyle = Light; 
  37.                     additionalContext = { 
  38.                         statusBarSettings = <XBStatusBarSettings: 0x600002c5c1a0; hidden: NO; style: 0x0; backgroundActivityEnabled: NO>; 
  39.                     } 
  40.                 } 
  41.                 imageContext = { 
  42.                     scale = 3.0; 
  43.                     opaque = YES; 
  44.                     fileRelativeLocation = default
  45.                     fileFormat = png; 
  46.                 } 
  47.             }; 
  48.             <XBApplicationSnapshot: 0x7fcb57004b60; identifier: 6B84614D-0867-4048-BE04-8E22E6742DDF; launchInterfaceIdentifier: __from_UILaunchStoryboardName__; contentType: GeneratedDefault; referenceSize: {375, 812}; interfaceOrientation: Portrait; userInterfaceStyle: Dark> { 
  49.                 creationDate = September 27, 2020 at 6:08:50 PM GMT+8; 
  50.                 keepsImageAccessUntilExpiration = NO
  51.                 hasGenerationContext = NO
  52.                 context = { 
  53.                     contentType = GeneratedDefault; 
  54.                     fullScreen = YES; 
  55.                     referenceSize = {375, 812}; 
  56.                     interfaceOrientation = Portrait; 
  57.                     userInterfaceStyle = Dark; 
  58.                     additionalContext = { 
  59.                         statusBarSettings = <XBStatusBarSettings: 0x600002c5c2d0; hidden: NO; style: 0x0; backgroundActivityEnabled: NO>; 
  60.                     } 
  61.                 } 
  62.                 imageContext = { 
  63.                     scale = 3.0; 
  64.                     opaque = YES; 
  65.                     fileRelativeLocation = default
  66.                     fileFormat = png; 
  67.                 } 
  68.             }; 
  69.             <XBApplicationSnapshot: 0x7fcb57005140; identifier: D3E8D00C-EE33-466B-98A6-7E60865D8001; launchInterfaceIdentifier: __from_UILaunchStoryboardName__; contentType: GeneratedDefault; referenceSize: {375, 812}; interfaceOrientation: LandscapeLeft; userInterfaceStyle: Light> { 
  70.                 creationDate = September 27, 2020 at 6:08:50 PM GMT+8; 
  71.                 keepsImageAccessUntilExpiration = NO
  72.                 hasGenerationContext = NO
  73.                 context = { 
  74.                     contentType = GeneratedDefault; 
  75.                     fullScreen = YES; 
  76.                     referenceSize = {375, 812}; 
  77.                     interfaceOrientation = LandscapeLeft; 
  78.                     userInterfaceStyle = Light; 
  79.                     additionalContext = { 
  80.                         statusBarSettings = <XBStatusBarSettings: 0x600002c5c400; hidden: YES; style: 0x0; backgroundActivityEnabled: NO>; 
  81.                     } 
  82.                 } 
  83.                 imageContext = { 
  84.                     scale = 3.0; 
  85.                     opaque = YES; 
  86.                     fileRelativeLocation = default
  87.                     fileFormat = png; 
  88.                 } 
  89.             }; 
  90.         }; 
  91.     } 

SplashBoard 部分类图

通过类信息,整理如下所示(只包含关键属性):

获取夜间模式启动图缓存路径

通常上面的内容,我们可以对 iOS 获取夜间模式启动图缓存路径的流程进行合理的猜测

启动时,会先通过 XBApplicationSnapshotGroup的identifier 获取test.SplashTest - {DEFAULT GROUP} 的启动图列表

再通过 XBApplicationSnapshot 的 userInterfaceStyle = Dark; 和 interfaceOrientation = Portrait;等信息,判断启动时应该使用

  1. <XBApplicationSnapshot: 0x7fcb57004b60; identifier: 6B84614D- 
  2. 0867-4048-BE04-8E22E6742DDF; launchInterfaceIdentifier:  
  3. __from_UILaunchStoryboardName__; contentType: GeneratedDefault;  
  4. referenceSize: {375, 812}; interfaceOrientation: Portrait;  
  5. userInterfaceStyle: Dark> 

最后再通过 XBApplicationSnapshot的 _relativePath 拼接启动图的真实路径

  1. ~/Library/Developer/CoreSimulator/Devices/1F9B22C5-E446-4881- 
  2. AFE4-3373E3513C59/data/Containers/Data/Application 
  3. /FA902232-17D2-495F-B23E-410349A9921C/Library/SplashBoard/Snapshots 
  4. /test.SplashTest - {DEFAULT GROUP}/6B84614D-0867-4048- 
  5. BE04-8E22E6742DDF@3x.ktx 

总结

本文通过对 applicationState.db 进行一系列的分析,最终实现了在模拟器下获取获取夜间模式启动图缓存路径的诉求。

 

责任编辑:未丽燕 来源: 酷酷的哀殿的博客
相关推荐

2021-06-28 14:35:36

iOSAPP缓存

2020-05-06 19:16:45

Windows 10Windows微软

2018-08-17 15:56:07

APP夜间模式理性思考

2021-05-06 06:22:36

手机夜间模式蓝光

2019-02-01 09:10:01

2017-03-31 14:31:08

QQ浏览器修复功能特性

2020-12-28 12:37:36

缓存击穿穿透

2021-08-26 17:36:42

Floyd算法数据结构

2018-06-22 13:25:14

2020-03-30 18:00:39

微信iOS应用移动应用

2021-10-30 20:49:24

算法 Xpath路径

2022-02-27 15:46:55

iOS省电模式技巧

2019-04-15 14:23:00

Windows 10Windows操作系统

2019-01-03 09:45:02

Ubuntu救援模式

2021-11-29 10:27:24

设计模式程序员

2009-06-18 15:23:49

缓存控制器模式Ajax模式

2011-07-25 16:50:24

Windows安全模式

2020-01-14 11:45:31

安全模式Windows 10Windows

2011-09-30 09:28:40

苹果谣言

2019-08-15 16:30:49

TomcatSpringBootJava
点赞
收藏

51CTO技术栈公众号