/*
*
* Apache License
* Version 2.0, January 2004
* http://www.apache.org/licenses/
*
* TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
*
* 1. Definitions.
*
* "License" shall mean the terms and conditions for use, reproduction,
* and distribution as defined by Sections 1 through 9 of this document.
*
* "Licensor" shall mean the copyright owner or entity authorized by
* the copyright owner that is granting the License.
*
* "Legal Entity" shall mean the union of the acting entity and all
* other entities that control, are controlled by, or are under common
* control with that entity. For the purposes of this definition,
* "control" means (i) the power, direct or indirect, to cause the
* direction or management of such entity, whether by contract or
* otherwise, or (ii) ownership of fifty percent (50%) or more of the
* outstanding shares, or (iii) beneficial ownership of such entity.
*
* "You" (or "Your") shall mean an individual or Legal Entity
* exercising permissions granted by this License.
*
* "Source" form shall mean the preferred form for making modifications,
* including but not limited to software source code, documentation
* source, and configuration files.
*
* "Object" form shall mean any form resulting from mechanical
* transformation or translation of a Source form, including but
* not limited to compiled object code, generated documentation,
* and conversions to other media types.
*
* "Work" shall mean the work of authorship, whether in Source or
* Object form, made available under the License, as indicated by a
* copyright notice that is included in or attached to the work
* (an example is provided in the Appendix below).
*
* "Derivative Works" shall mean any work, whether in Source or Object
* form, that is based on (or derived from) the Work and for which the
* editorial revisions, annotations, elaborations, or other modifications
* represent, as a whole, an original work of authorship. For the purposes
* of this License, Derivative Works shall not include works that remain
* separable from, or merely link (or bind by name) to the interfaces of,
* the Work and Derivative Works thereof.
*
* "Contribution" shall mean any work of authorship, including
* the original version of the Work and any modifications or additions
* to that Work or Derivative Works thereof, that is intentionally
* submitted to Licensor for inclusion in the Work by the copyright owner
* or by an individual or Legal Entity authorized to submit on behalf of
* the copyright owner. For the purposes of this definition, "submitted"
* means any form of electronic, verbal, or written communication sent
* to the Licensor or its representatives, including but not limited to
* communication on electronic mailing lists, source code control systems,
* and issue tracking systems that are managed by, or on behalf of, the
* Licensor for the purpose of discussing and improving the Work, but
* excluding communication that is conspicuously marked or otherwise
* designated in writing by the copyright owner as "Not a Contribution."
*
* "Contributor" shall mean Licensor and any individual or Legal Entity
* on behalf of whom a Contribution has been received by Licensor and
* subsequently incorporated within the Work.
*
* 2. Grant of Copyright License. Subject to the terms and conditions of
* this License, each Contributor hereby grants to You a perpetual,
* worldwide, non-exclusive, no-charge, royalty-free, irrevocable
* copyright license to reproduce, prepare Derivative Works of,
* publicly display, publicly perform, sublicense, and distribute the
* Work and such Derivative Works in Source or Object form.
*
* 3. Grant of Patent License. Subject to the terms and conditions of
* this License, each Contributor hereby grants to You a perpetual,
* worldwide, non-exclusive, no-charge, royalty-free, irrevocable
* (except as stated in this section) patent license to make, have made,
* use, offer to sell, sell, import, and otherwise transfer the Work,
* where such license applies only to those patent claims licensable
* by such Contributor that are necessarily infringed by their
* Contribution(s) alone or by combination of their Contribution(s)
* with the Work to which such Contribution(s) was submitted. If You
* institute patent litigation against any entity (including a
* cross-claim or counterclaim in a lawsuit) alleging that the Work
* or a Contribution incorporated within the Work constitutes direct
* or contributory patent infringement, then any patent licenses
* granted to You under this License for that Work shall terminate
* as of the date such litigation is filed.
*
* 4. Redistribution. You may reproduce and distribute copies of the
* Work or Derivative Works thereof in any medium, with or without
* modifications, and in Source or Object form, provided that You
* meet the following conditions:
*
* (a) You must give any other recipients of the Work or
* Derivative Works a copy of this License; and
*
* (b) You must cause any modified files to carry prominent notices
* stating that You changed the files; and
*
* (c) You must retain, in the Source form of any Derivative Works
* that You distribute, all copyright, patent, trademark, and
* attribution notices from the Source form of the Work,
* excluding those notices that do not pertain to any part of
* the Derivative Works; and
*
* (d) If the Work includes a "NOTICE" text file as part of its
* distribution, then any Derivative Works that You distribute must
* include a readable copy of the attribution notices contained
* within such NOTICE file, excluding those notices that do not
* pertain to any part of the Derivative Works, in at least one
* of the following places: within a NOTICE text file distributed
* as part of the Derivative Works; within the Source form or
* documentation, if provided along with the Derivative Works; or,
* within a display generated by the Derivative Works, if and
* wherever such third-party notices normally appear. The contents
* of the NOTICE file are for informational purposes only and
* do not modify the License. You may add Your own attribution
* notices within Derivative Works that You distribute, alongside
* or as an addendum to the NOTICE text from the Work, provided
* that such additional attribution notices cannot be construed
* as modifying the License.
*
* You may add Your own copyright statement to Your modifications and
* may provide additional or different license terms and conditions
* for use, reproduction, or distribution of Your modifications, or
* for any such Derivative Works as a whole, provided Your use,
* reproduction, and distribution of the Work otherwise complies with
* the conditions stated in this License.
*
* 5. Submission of Contributions. Unless You explicitly state otherwise,
* any Contribution intentionally submitted for inclusion in the Work
* by You to the Licensor shall be under the terms and conditions of
* this License, without any additional terms or conditions.
* Notwithstanding the above, nothing herein shall supersede or modify
* the terms of any separate license agreement you may have executed
* with Licensor regarding such Contributions.
*
* 6. Trademarks. This License does not grant permission to use the trade
* names, trademarks, service marks, or product names of the Licensor,
* except as required for reasonable and customary use in describing the
* origin of the Work and reproducing the content of the NOTICE file.
*
* 7. Disclaimer of Warranty. Unless required by applicable law or
* agreed to in writing, Licensor provides the Work (and each
* Contributor provides its Contributions) on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied, including, without limitation, any warranties or conditions
* of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
* PARTICULAR PURPOSE. You are solely responsible for determining the
* appropriateness of using or redistributing the Work and assume any
* risks associated with Your exercise of permissions under this License.
*
* 8. Limitation of Liability. In no event and under no legal theory,
* whether in tort (including negligence), contract, or otherwise,
* unless required by applicable law (such as deliberate and grossly
* negligent acts) or agreed to in writing, shall any Contributor be
* liable to You for damages, including any direct, indirect, special,
* incidental, or consequential damages of any character arising as a
* result of this License or out of the use or inability to use the
* Work (including but not limited to damages for loss of goodwill,
* work stoppage, computer failure or malfunction, or any and all
* other commercial damages or losses), even if such Contributor
* has been advised of the possibility of such damages.
*
* 9. Accepting Warranty or Additional Liability. While redistributing
* the Work or Derivative Works thereof, You may choose to offer,
* and charge a fee for, acceptance of support, warranty, indemnity,
* or other liability obligations and/or rights consistent with this
* License. However, in accepting such obligations, You may act only
* on Your own behalf and on Your sole responsibility, not on behalf
* of any other Contributor, and only if You agree to indemnify,
* defend, and hold each Contributor harmless for any liability
* incurred by, or claims asserted against, such Contributor by reason
* of your accepting any such warranty or additional liability.
*
* END OF TERMS AND CONDITIONS
*
* APPENDIX: How to apply the Apache License to your work.
*
* To apply the Apache License to your work, attach the following
* boilerplate notice, with the fields enclosed by brackets "[]"
* replaced with your own identifying information. (Don't include
* the brackets!) The text should be enclosed in the appropriate
* comment syntax for the file format. We also recommend that a
* file or class name and description of purpose be included on the
* same "printed page" as the copyright notice for easier
* identification within third-party archives.
*
* Copyright 2016 Alibaba Group
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*
*
*/
package android.taobao.atlas.bundleInfo;
import android.content.ComponentName;
import android.content.Context;
import android.content.pm.ActivityInfo;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.ProviderInfo;
import android.content.pm.ServiceInfo;
import android.taobao.atlas.runtime.RuntimeVariables;
import android.taobao.atlas.util.log.impl.AtlasMonitor;
import android.text.TextUtils;
import java.io.File;
import java.util.*;
/**
* Created by guanjie on 15/7/6.
*/
public class AtlasBundleInfoManager {
public static final String TAG = "AtlasBundleInfoManager";
private static AtlasBundleInfoManager sManager;
private final String LIST_FILE_DIR ;
private final String BUNDLE_LIST_FILE_PREFIX = "bundleInfo-";
private BundleListing mCurrentBundleListing;
private static final String CHARSET = "UTF-8";
//
private static HashMap<String,ActivityInfo> activityInfos;
private static HashMap<String,ServiceInfo> serviceInfos;
private static HashMap<String,ActivityInfo> receiverInfos;
private static HashMap<String,ProviderInfo> providerInfos;
private static HashMap<String,SimpleBundleInfo> bundleBaseInfos = new HashMap<String, SimpleBundleInfo>();
public static synchronized AtlasBundleInfoManager instance(){
if(sManager==null){
sManager = new AtlasBundleInfoManager();
}
return sManager;
}
private AtlasBundleInfoManager(){
Context context = RuntimeVariables.androidApplication;
LIST_FILE_DIR = context.getFilesDir().getAbsolutePath()+ File.separatorChar+"bundlelisting"+File.separatorChar;
File file = new File(LIST_FILE_DIR);
if(!file.exists()){
file.mkdirs();
}
}
public BundleListing getBundleInfo(){
InitBundleInfoByVersionIfNeed();
return mCurrentBundleListing;
}
/*
* Get all dependent bundles for the designated bundle
*/
public List<String> getDependencyForBundle(String bundleName){
InitBundleInfoByVersionIfNeed();
if (mCurrentBundleListing == null || mCurrentBundleListing.getBundles()==null || mCurrentBundleListing.getBundles().size() ==0){
getBundleInfoFromManifestIfNeed();
SimpleBundleInfo info = bundleBaseInfos.get(bundleName);
if(info!=null) {
return info.dependency;
}else{
return null;
}
}
BundleListing.BundleInfo bundleInfo = mCurrentBundleListing.getBundles().get(bundleName);
if(bundleInfo != null && bundleInfo.getDependency() != null){
List<String> dependencies = new ArrayList<String>();
for(int x=0; x<bundleInfo.getDependency().size();x++){
if(!TextUtils.isEmpty(bundleInfo.getDependency().get(x))){
dependencies.add(bundleInfo.getDependency().get(x));
}
}
return dependencies;
}
return null;
}
public boolean isInternalBundle(String bundleName){
InitBundleInfoByVersionIfNeed();
if (mCurrentBundleListing == null || mCurrentBundleListing.getBundles()==null || mCurrentBundleListing.getBundles().size() ==0){
getBundleInfoFromManifestIfNeed();
SimpleBundleInfo sinfo = bundleBaseInfos.get(bundleName);
if(sinfo!=null) {
return sinfo.isInternal;
}else{
return true;
}
}else{
BundleListing.BundleInfo info = mCurrentBundleListing.getBundles().get(bundleName);
if(info!=null){
return info.isInternal();
}else{
return true;
}
}
}
/*
* Get all dependent bundles for the designated bundle
*/
public boolean getHasSO(String bundleName){
InitBundleInfoByVersionIfNeed();
if (mCurrentBundleListing == null || mCurrentBundleListing.getBundles()==null || mCurrentBundleListing.getBundles().size() == 0){
return false;
}
BundleListing.BundleInfo info = mCurrentBundleListing.getBundles().get(bundleName);
if(info!=null){
return info.isHasSO();
}
return false;
}
public String getBundleForComponet(String componentName){
InitBundleInfoByVersionIfNeed();
if (mCurrentBundleListing == null ||
mCurrentBundleListing.getBundles() == null ||
mCurrentBundleListing.getBundles().size() == 0){
return findBundleByComponentName(componentName);
}
Iterator<Map.Entry<String, BundleListing.BundleInfo>> iterator = mCurrentBundleListing.getBundles().entrySet().iterator();
while (iterator.hasNext()) {
Map.Entry<String, BundleListing.BundleInfo> entry = iterator.next();
BundleListing.BundleInfo bundleInfo = entry.getValue();
if(bundleInfo!=null && bundleInfo.getActivities()!=null && bundleInfo.getActivities().containsKey(componentName)) {
return bundleInfo.getPkgName();
}
if(bundleInfo!=null && bundleInfo.getServices()!=null && bundleInfo.getServices().containsKey(componentName)) {
return bundleInfo.getPkgName();
}
if(bundleInfo!=null && bundleInfo.getReceivers()!=null && bundleInfo.getReceivers().containsKey(componentName)) {
return bundleInfo.getPkgName();
}
if(bundleInfo!=null && bundleInfo.getContentProviders()!=null && bundleInfo.getContentProviders().containsKey(componentName)) {
return bundleInfo.getPkgName();
}
}
return null;
}
/*
* Return designated bundle info
*/
public BundleListing.BundleInfo getBundleInfo(String name){
InitBundleInfoByVersionIfNeed();
if (mCurrentBundleListing == null || mCurrentBundleListing.getBundles()==null || mCurrentBundleListing.getBundles().size() ==0){
getBundleInfoFromManifestIfNeed();
SimpleBundleInfo sinfo = bundleBaseInfos.get(name);
BundleListing.BundleInfo info= new BundleListing.BundleInfo();
info.setDependency(sinfo!=null ? sinfo.dependency : null);
info.setName("动态模块");
info.setApplicationName(sinfo!=null ? sinfo.applicationName : "");
return info;
}
BundleListing.BundleInfo info = mCurrentBundleListing.getBundles().get(name);
if(info!=null){
return info;
}
return null;
}
// public void persistListToFile (final BundleListing listing,final String version){
// if (null == listing){
// return;
// }
// if(Thread.currentThread().getId() == Looper.getMainLooper().getThread().getId()){
// new AsyncTask<String, String, String>() {
// @Override
// protected String doInBackground(String... params) {
// persistListToFileInternal(listing, version);
// return "";
// }
// }.execute();
// }else{
// persistListToFileInternal(listing,version);
// }
//
// }
// public void persistListToFileInternal(final BundleListing listing,final String version){
// String content= BundleListingUtil.toJSONString(listing.getBundles());
// Log.d(TAG, "new listing = " + content);
// String fileName = String.format("%s%s.json",BUNDLE_LIST_FILE_PREFIX,version);
// File bundleInfoFile = new File(String.format("%s%s",LIST_FILE_DIR,fileName));
// for (int i = 0;i <3;i++){
// if(!bundleInfoFile.exists()){
// try {
// bundleInfoFile.createNewFile();
// }catch (IOException e){
// e.printStackTrace();
// }
// }
// }
// try {
// BufferedWriter bufferWritter = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(bundleInfoFile),CHARSET));
// bufferWritter.write(content);
// bufferWritter.close();
// } catch (IOException e) {
// e.printStackTrace();
// }
// }
/**
* 更新当前清单
* @param updateBundles 更新的bundle列表
* @param newVersion 主客新版本
*/
// public void saveNewBundlelisingWithMerge(List<BundleListing.BundleInfo> updateBundles,String newVersion){
// Log.d(TAG,"mergeCurrentListWithUpdate");
// InitBundleInfoByVersionIfNeed();
// BundleListing cloneListing = BundleListing.clone(mCurrentBundleListing);
// if(cloneListing==null || cloneListing.getBundles()==null){
// Log.e("BundleInfoManager","list is null is impossible");
// return;
// }
// Log.d(TAG,"old listing size = "+ cloneListing.getBundles().size());
//
// for(BundleListing.BundleInfo updateInfo : updateBundles){
// BundleListing.BundleInfo oldInfo = cloneListing.getBundles().get(updateInfo.getPkgName());
// if(oldInfo!=null){
// oldInfo.setVersion(updateInfo.getVersion());
// oldInfo.setMd5(updateInfo.getMd5());
// oldInfo.setSize(updateInfo.getSize());
// oldInfo.setDependency(updateInfo.getDependency());
// oldInfo.setUrl(updateInfo.getUrl());
// }else{
// BundleListing.BundleInfo info = new BundleListing.BundleInfo();
// info.setPkgName(updateInfo.getPkgName());
// info.setVersion(updateInfo.getVersion());
// info.setMd5(updateInfo.getMd5());
// info.setSize(updateInfo.getSize());
// info.setDependency(updateInfo.getDependency());
// info.setUrl(updateInfo.getUrl());
// cloneListing.insertBundle(info);
// }
// }
// persistListToFile(cloneListing,newVersion);
// }
/**
* 根据版本载入清单
*/
private synchronized void InitBundleInfoByVersionIfNeed(){
if(mCurrentBundleListing==null){
String bundleInfoStr = (String)RuntimeVariables.getFrameworkProperty("bundleInfo");
if(!TextUtils.isEmpty(bundleInfoStr)) {
try {
LinkedHashMap<String,BundleListing.BundleInfo> infos = BundleListingUtil.parseArray(bundleInfoStr);
if (infos == null) {
AtlasMonitor.getInstance().trace(AtlasMonitor.CONTAINER_BUNDLEINFO_PARSE_FAIL,
false, "0", "", bundleInfoStr);
}
BundleListing listing = new BundleListing();
listing.setBundles(infos);
mCurrentBundleListing = listing;
}catch(Throwable e){
e.printStackTrace();
}
}else{
throw new RuntimeException("read bundleInfo failed");
}
}
}
// private String getFromAssets(String fileName,Context context){
// BufferedReader bufReader = null;
// try {
// InputStreamReader inputReader = new InputStreamReader(context.getResources().getAssets().open(fileName), CHARSET);
// bufReader = new BufferedReader(inputReader);
// String line="";
// String result="";
// while((line = bufReader.readLine()) != null)
// result += line;
// return result;
// } catch (Exception e) {
// e.printStackTrace();
// return null;
// } finally {
// if(bufReader!=null){
// try {
// bufReader.close();
// } catch (IOException e) {
// e.printStackTrace();
// }
// }
// }
// }
//
// private String getFromFile(String fileName){
// File file = new File(fileName);
// Long fileLength = file.length();
// byte[] filecontent = new byte[fileLength.intValue()];
// try {
// FileInputStream in = new FileInputStream(file);
// in.read(filecontent);
// in.close();
// return new String(filecontent,CHARSET);
// } catch (Throwable e) {
// e.printStackTrace();
// }
// return null;
// }
private String findBundleByComponentName(String componentClassName){
getComponentInfoFromManifestIfNeed();
ComponentName componentName = new ComponentName(RuntimeVariables.androidApplication.getPackageName(),componentClassName);
if(activityInfos!=null){
ActivityInfo info = activityInfos.get(componentClassName);
if(info!=null){
if(info.metaData!=null){
return info.metaData.getString("bundleLocation");
}else {
try {
ActivityInfo detailInfo = RuntimeVariables.androidApplication.getPackageManager().getActivityInfo(componentName, PackageManager.GET_META_DATA);
if (detailInfo != null && detailInfo.metaData != null) {
info.metaData = detailInfo.metaData;
return detailInfo.metaData.getString("bundleLocation");
} else {
return null;
}
} catch (Throwable e) {
}
}
}
}
if(serviceInfos!=null){
ServiceInfo info = serviceInfos.get(componentClassName);
if(info!=null) {
if (info.metaData != null) {
return info.metaData.getString("bundleLocation");
} else {
try {
ServiceInfo detailInfo = RuntimeVariables.androidApplication.getPackageManager().getServiceInfo(componentName, PackageManager.GET_META_DATA);
if (detailInfo != null && detailInfo.metaData != null) {
info = detailInfo;
return detailInfo.metaData.getString("bundleLocation");
} else {
return null;
}
} catch (Throwable e) {
}
}
}
}
if(receiverInfos!=null){
ActivityInfo info = receiverInfos.get(componentClassName);
if(info!=null){
if(info.metaData!=null){
return info.metaData.getString("bundleLocation");
}else {
try {
ActivityInfo detailInfo = RuntimeVariables.androidApplication.getPackageManager().getReceiverInfo(componentName,PackageManager.GET_META_DATA);
if(detailInfo!=null && detailInfo.metaData!=null){
info.metaData = detailInfo.metaData;
return detailInfo.metaData.getString("bundleLocation");
}else{
return null;
}
} catch (Throwable e) {}
}
}
}
if(providerInfos!=null){
ProviderInfo info = providerInfos.get(componentClassName);
if(info!=null){
if(info.metaData!=null){
return info.metaData.getString("bundleLocation");
}else {
try {
ProviderInfo detailInfo = RuntimeVariables.androidApplication.getPackageManager().getProviderInfo(componentName,PackageManager.GET_META_DATA);
if(detailInfo!=null && detailInfo.metaData!=null){
info.metaData = detailInfo.metaData;
return detailInfo.metaData.getString("bundleLocation");
}else{
return null;
}
} catch (Throwable e) {}
}
}
}
return null;
}
private synchronized void getComponentInfoFromManifestIfNeed(){
try {
if(activityInfos==null) {
PackageInfo infos = RuntimeVariables.androidApplication.getPackageManager().getPackageInfo(RuntimeVariables.androidApplication.getPackageName(), PackageManager.GET_ACTIVITIES);
if(infos.activities!=null) {
activityInfos = new HashMap<String, ActivityInfo>();
for (ActivityInfo info : infos.activities) {
activityInfos.put(info.name,info);
}
}
}
if(serviceInfos==null) {
PackageInfo infos = RuntimeVariables.androidApplication.getPackageManager().getPackageInfo(RuntimeVariables.androidApplication.getPackageName(), PackageManager.GET_SERVICES);
if(infos.services!=null) {
serviceInfos = new HashMap<String, ServiceInfo>();
for (ServiceInfo info : infos.services) {
serviceInfos.put(info.name,info);
}
}
}
if(receiverInfos==null) {
PackageInfo infos = RuntimeVariables.androidApplication.getPackageManager().getPackageInfo(RuntimeVariables.androidApplication.getPackageName(), PackageManager.GET_RECEIVERS);
if(infos.receivers!=null) {
receiverInfos = new HashMap<String, ActivityInfo>();
for (ActivityInfo info : infos.receivers) {
receiverInfos.put(info.name,info);
}
}
}
if(providerInfos==null) {
PackageInfo infos = RuntimeVariables.androidApplication.getPackageManager().getPackageInfo(RuntimeVariables.androidApplication.getPackageName(), PackageManager.GET_PROVIDERS);
if(infos.providers!=null) {
providerInfos = new HashMap<String, ProviderInfo>();
for (ProviderInfo info : infos.providers) {
providerInfos.put(info.name,info);
}
}
}
}catch(Throwable e){
}
}
private synchronized void getBundleInfoFromManifestIfNeed(){
if(bundleBaseInfos.size()!=0){
return;
}
try{
try {
ApplicationInfo appInfo = RuntimeVariables.androidApplication.getPackageManager()
.getApplicationInfo(RuntimeVariables.androidApplication.getPackageName(),PackageManager.GET_META_DATA);
if(appInfo.metaData!=null){
Set<String> keySet = appInfo.metaData.keySet();
for(String key : keySet){
if(key.startsWith("bundle_")){
String bundleName = key.substring(7);
String value = appInfo.metaData.getString(key);
String[] result = value.split(",");
SimpleBundleInfo info = new SimpleBundleInfo();
info.applicationName = result[0];
info.isInternal = Boolean.valueOf(result[1]);
if(result.length>2){
String[] dependency = result[1].split("\\|");
if(dependency!=null && dependency.length>0){
info.dependency = Arrays.asList(dependency);
}
}
bundleBaseInfos.put(bundleName,info);
}
}
}
} catch (PackageManager.NameNotFoundException e) {
e.printStackTrace();
}
}catch(Throwable e){
}
}
private static class SimpleBundleInfo{
public String applicationName;
public boolean isInternal;
public List<String> dependency;
}
}