OpenHarmony开发者论坛

标题: Launcher应用安装问题分析报告 [打印本页]

作者: Laval社区小助手    时间: 2024-4-1 10:14
标题: Launcher应用安装问题分析报告
[md]# 1 关键字

Launcher;Storage

# 2 问题描述

问题现象:安装应用到达20个以上后,重启设备,Launcher页面没有任何应用图标

运行环境:硬件 dayu200,软件:3.1release

测试步骤:

1. 使用hdc将测试安装包安装到设备上,安装数量在20个以上,尽可能稍多一些
2. 重启设备
3. Launcher桌面显示异常,不显示任何应用图标

# 3 问题原因

## 3.1 正常机制

* Launcher的桌面应用布局信息是通过Storage存储在文件中。
* Launcher在系统安装应用后会更新桌面布局配置,调用putSync将当前布局写入内存中,然后通过flushSync持久化到文件中。
* 桌面布局配置持久化后,设备重启,Launcher页面正常加载。

## 3.2 异常机制

* 当桌面应用安装过多,桌面布局配置信息超过8192字节时,底层通过定长数组接收参数,导致数组越界未定义,从而无法正确获取桌面布局配置信息,属性为空,从而将内存中属性更新为空了。
* 持久化过程中再将内存中的属性更新到文件中,从而导致文件信息也被更新为空。
* 重启设备,重新加载文件,桌面布局配置为空,从而无法显示任何应用图标。

[Storage官方文档](https://link.csdn.net/?target=ht ... 3Flogin%3Dfrom_csdn)说明:key的最大长度限制,需小于80字节。value的最大长度限制,需小于8192字节。

# 4 解决方案

* 应用层面,可通过对超长字段进行拆分存储,规避该问题
* 存储方式,可通过使用关系型数据库@ohos.data.rdb的rdbStore或分布式数据管理@ohos.data.distributedData的kvStore来进行数据存储
* 底层实现,系统目前文件存储的最大长度限制为8192字节,可通过修改底层存储逻辑,修改限定长度或自动扩容实现

# 5 定位过程

该问题为底层实现与特殊场景不兼容导致的运行异常:

* 安装应用20+复现相关问题。
* 关注系统桌面布局配置LauncherPreference文件中的DesktopApplicationInfo信息更新情况。
  文件路径:/data/app/el2/100/database/com.ohos.launcher/pad/LauncherPreference
  文件内容:

  ```
  <?xml version="1.0" encoding="UTF-8"?>
  <preferences version="1.0">
    <string key="DesktopApplicationInfo">[{"appName":"$string:MainAbility_label","isSystemApp":true,"isUninstallAble":false,"appIconId":16777219,"appLabelId":16777217,"bundleName":"com.example.AircraftWar","abilityName":"com.example.AircraftWar.MainAbility","type":0,"area":[1,1]},{"appName":"$string:MainAbility_label","isSystemApp":true,"isUninstallAble":false,"appIconId":16777219,"appLabelId":16777217,"bundleName":"com.example.Billiards","abilityName":"com.example.Billiards.MainAbility","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":true,"isUninstallAble":false,"appIconId":16777229,"appLabelId":16777219,"bundleName":"com.example.Browser","abilityName":"MainAbility","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":false,"isUninstallAble":true,"appIconId":16777219,"appLabelId":16777216,"bundleName":"com.example.baseanimation","abilityName":"com.example.baseanimation.MainAbility","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":true,"isUninstallAble":false,"appIconId":16777218,"appLabelId":16777216,"bundleName":"com.example.distributedcalc","abilityName":"com.example.distributedcalc.default","type":0,"area":[1,1]},{"appName":"$string:entry_MainAbility","isSystemApp":true,"isUninstallAble":false,"appIconId":16777218,"appLabelId":16777217,"bundleName":"com.example.shopping","abilityName":"com.example.shopping.MainAbility","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":false,"isUninstallAble":true,"appIconId":16777217,"appLabelId":16777219,"bundleName":"com.huawei.himovie","abilityName":"MainAbility","type":0,"area":[1,1]},{"appName":"$string:entry_Music","isSystemApp":true,"isUninstallAble":false,"appIconId":16777311,"appLabelId":16777219,"bundleName":"com.huawei.himusicdemo","abilityName":"com.huawei.himusicdemo.Music","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":true,"isUninstallAble":false,"appIconId":150995030,"appLabelId":150994944,"bundleName":"com.ohos.camera","abilityName":"com.ohos.camera.MainAbility","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":true,"isUninstallAble":false,"appIconId":16777431,"appLabelId":16777225,"bundleName":"com.ohos.contacts","abilityName":"com.ohos.contacts.MainAbility","type":0,"area":[1,1]},{"appName":"$string:messages","isSystemApp":true,"isUninstallAble":false,"appIconId":16777565,"appLabelId":16777321,"bundleName":"com.ohos.mms","abilityName":"com.ohos.mms.MainAbility","type":0,"area":[1,1]},{"appName":"$string:entry_MainAbility","isSystemApp":true,"isUninstallAble":false,"appIconId":134217900,"appLabelId":134217748,"bundleName":"com.ohos.note","abilityName":"com.ohos.note.MainAbility","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":true,"isUninstallAble":false,"appIconId":16777834,"appLabelId":16777216,"bundleName":"com.ohos.photos","abilityName":"com.ohos.photos.MainAbility","type":0,"area":[1,1]},{"appName":"$string:entry_MainAbility","isSystemApp":true,"isUninstallAble":false,"appIconId":50332153,"appLabelId":50331728,"bundleName":"com.ohos.settings","abilityName":"com.ohos.settings.MainAbility","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":true,"isUninstallAble":false,"appIconId":16777218,"appLabelId":16777216,"bundleName":"ohos.samples.airquality","abilityName":"ohos.samples.airquality.default","type":0,"area":[1,1]},{"appName":"$string:app_name","isSystemApp":true,"isUninstallAble":false,"appIconId":16777218,"appLabelId":16777216,"bundleName":"ohos.samples.clock","abilityName":"ohos.samples.clock.default","type":0,"area":[1,1]},{"appName":"$string:entry_MainAbility","isSystemApp":true,"isUninstallAble":false,"appIconId":16777219,"appLabelId":16777217,"bundleName":"ohos.samples.webdemo","abilityName":"ohos.samples.webdemo.MainAbility","type":0,"area":[1,1]}]</string>
    <string key="DesktopModeConfig">{"appStartPageType":"Grid","gridConfig":0,"deviceType":"pad"}</string>
    <string key="GridLayoutInfo">{"layoutDescription":{"pageCount":1,"row":5,"column":11},"layoutInfo":[{"bundleName":"com.example.AircraftWar","type":0,"area":[1,1],"page":0,"column":0,"row":0},{"bundleName":"com.example.Billiards","type":0,"area":[1,1],"page":0,"column":1,"row":0},{"bundleName":"com.example.Browser","type":0,"area":[1,1],"page":0,"column":2,"row":0},{"bundleName":"com.example.baseanimation","type":0,"area":[1,1],"page":0,"column":3,"row":0},{"bundleName":"com.example.distributedcalc","type":0,"area":[1,1],"page":0,"column":4,"row":0},{"bundleName":"com.example.shopping","type":0,"area":[1,1],"page":0,"column":5,"row":0},{"bundleName":"com.huawei.himovie","type":0,"area":[1,1],"page":0,"column":6,"row":0},{"bundleName":"com.huawei.himusicdemo","type":0,"area":[1,1],"page":0,"column":7,"row":0},{"bundleName":"com.ohos.camera","type":0,"area":[1,1],"page":0,"column":8,"row":0},{"bundleName":"com.ohos.contacts","type":0,"area":[1,1],"page":0,"column":9,"row":0},{"bundleName":"com.ohos.mms","type":0,"area":[1,1],"page":0,"column":10,"row":0},{"bundleName":"com.ohos.note","type":0,"area":[1,1],"page":0,"column":0,"row":1},{"bundleName":"com.ohos.photos","type":0,"area":[1,1],"page":0,"column":1,"row":1},{"bundleName":"com.ohos.settings","type":0,"area":[1,1],"page":0,"column":2,"row":1},{"bundleName":"ohos.samples.airquality","type":0,"area":[1,1],"page":0,"column":3,"row":1},{"bundleName":"ohos.samples.clock","type":0,"area":[1,1],"page":0,"column":4,"row":1},{"bundleName":"ohos.samples.webdemo","type":0,"area":[1,1],"page":0,"column":5,"row":1}]}</string>
    <string key="SmartDockLayoutInfo">[{"itemType":2,"editable":false,"bundleName":"com.ohos.launcher","abilityName":"com.ohos.launcher.appcenter.MainAbility","appIconId":184549428,"appLabelId":184549400,"appName":"全部应用"},{"itemType":2,"editable":false,"bundleName":"com.ohos.launcher","abilityName":"com.ohos.launcher.recents.MainAbility","appIconId":184549429,"appLabelId":184549401,"appName":"最近任务"},{"itemType":0,"editable":true,"appName":"图库","bundleName":"com.ohos.photos","abilityName":"com.ohos.photos.MainAbility","appIconId":16777834,"appLabelId":16777216},{"itemType":0,"editable":false,"appName":"设置","bundleName":"com.ohos.settings","abilityName":"com.ohos.settings.MainAbility","appIconId":50332153,"appLabelId":50331728}]</string>
  </preferences>
  ```
* 关注属性写入内存putSync与内存信息持久化到文件flushSync调用前后的属性变量。
* 通过日志定位到问题原因:putSync调用导致的异常。
* 分析api底层实现:底层实现调用napi\_storage.cpp的StorageProxy::SetValueSync。
* SetValueSync具体逻辑:声明定长char数组用来接收key(MAX\_KEY\_LENGTH = 80),获取value类型,当类型为string时,声明定长char数组用来接收value(MAX\_VALUE\_LENGTH = 8 \* 1024),然后调用preferences\_impl.cpp的PutString。

  ```
  napi_value StorageProxy::SetValueSync(napi_env env, napi_callback_info info)
  {
      napi_value thiz = nullptr;
      size_t argc = 2;
      napi_value args[2] = { 0 };

      LOG_DEBUG("SETVALUE");
      NAPI_CALL(env, napi_get_cb_info(env, info, &argc, args, &thiz, nullptr));
      NAPI_ASSERT(env, argc == 2, "Wrong number of arguments");
      // get value type
      napi_valuetype valueType = napi_undefined;
      NAPI_CALL(env, napi_typeof(env, args[0], &valueType));
      NAPI_ASSERT(env, valueType == napi_string, "type mismatch for key");

      // get input key
      char key[MAX_KEY_LENGTH] = { 0 };
      size_t out = 0;
      NAPI_CALL(env, napi_get_value_string_utf8(env, args[0], key, MAX_KEY_LENGTH, &out));

      StorageProxy *obj = nullptr;
      NAPI_CALL(env, napi_unwrap(env, thiz, reinterpret_cast<void **>(&obj)));
      NAPI_ASSERT(env, (obj != nullptr && obj->value_ != nullptr), "unwrap null native pointer");

      NAPI_CALL(env, napi_typeof(env, args[1], &valueType));
      if (valueType == napi_number) {
          double value = 0.0;
          NAPI_CALL(env, napi_get_value_double(env, args[1], &value));
          int result = obj->value_->utDouble(key, (double)value);
          NAPI_ASSERT(env, result == E_OK, "call PutDouble failed");
      } else if (valueType == napi_string) {
          char *value = new char[MAX_VALUE_LENGTH];

          napi_status status = napi_get_value_string_utf8(env, args[1], value, MAX_VALUE_LENGTH, &out);
          if (status != napi_ok) {
              LOG_DEBUG("napi_get_value_string_utf8 failed");
              LOG_DEBUG(value);
          } else {
              // get value
              int result = obj->value_->utString(key, value);
          }        
          delete[] value;
          NAPI_ASSERT(env, result == E_OK, "call PutString failed");
      } else if (valueType == napi_boolean) {
          bool value = false;
          NAPI_CALL(env, napi_get_value_bool(env, args[1], &value));
          // get value
          int result = obj->value_->utBool(key, value);
          NAPI_ASSERT(env, result == E_OK, "call PutBool failed");
      } else {
          NAPI_ASSERT(env, false, "Wrong second parameter type");
      }
      return nullptr;
  }
  ```

  **注:当属性长度超长后会导致数组越界未定义,从而无法正常获取属性值,导致属性值为空,从而导致内存中存放属性有误**
* PutString具体逻辑:调用CheckStringValue进行value值校验(校验长度),校验通过调用PutPreferencesValue。

  ```
  int PreferencesImpl:utString(const std::string &key, const std::string &value)
  {
      int errCode = CheckKey(key);
      if (errCode != E_OK) {
          return errCode;
      }
      errCode = CheckStringValue(value);
      if (errCode != E_OK) {
          return errCode;
      }
      PutPreferencesValue(key, PreferencesValue(value));
      return E_OK;
  }
  ```

  ```
  int PreferencesImpl::CheckKey(const std::string &key)
  {
      if (key.empty()) {
          LOG_ERROR("The key string is null or empty.");
          return E_KEY_EMPTY;
      }
      if (Preferences::MAX_KEY_LENGTH < key.length()) {
          LOG_ERROR("The key string length should shorter than 80.");
          return E_KEY_EXCEED_MAX_LENGTH;
      }
      return E_OK;
  }
  ```

  ```
  int PreferencesImpl::CheckStringValue(const std::string &value)
  {
      if (Preferences::MAX_VALUE_LENGTH < value.length()) {
          LOG_ERROR("The value string length should shorter than 8 * 1024.");
          return E_VALUE_EXCEED_MAX_LENGTH;
      }
      return E_OK;
  }
  ```
* PutPreferencesValue具体逻辑:AwaitLoadFile获取资源锁,调用insert\_or\_assign存放属性到内存map中,调用push\_back记录已修改属性的key值到内存map中。

  ```
  void PreferencesImpl:utPreferencesValue(const std::string &key, const PreferencesValue &value)
  {
      AwaitLoadFile();

      std::lock_guard<std::mutex> lock(mutex_);

      auto iter = map_.find(key);
      if (iter != map_.end()) {
          PreferencesValue &val = iter->second;
          if (val == value) {
              return;
          }
      }

      map_.insert_or_assign(key, value);
      modifiedKeys_.push_back(key);
  }
  ```
* 属性写入内存后,再由flushSync来进行数据的持久化操作,将内存数据写入到文件中。

# 6 知识分享

* OpenHarmony文件存储@ohos.data.storage目前的长度限制:key的最大长度限制,需小于80字节,value的最大长度限制,需小于8192字节。
* 数据库存储目前逻辑上是没有长度限制的,长度限制需参考数据库属性配置,sqlite中text存储可变长度的非Unicode数据,最大长度为2^31-1(2,147,483,647)个字符。
[/md]




欢迎光临 OpenHarmony开发者论坛 (https://forums.openharmony.cn/) Powered by Discuz! X3.5