package com.fox2code.mmm.repo; import android.content.SharedPreferences; import com.fox2code.mmm.BuildConfig; import com.fox2code.mmm.MainApplication; import com.fox2code.mmm.R; import com.fox2code.mmm.manager.ModuleInfo; import com.fox2code.mmm.utils.Files; import com.fox2code.mmm.utils.PropUtils; import org.json.JSONArray; import org.json.JSONException; import org.json.JSONObject; import java.io.File; import java.io.IOException; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.HashMap; import java.util.Iterator; import java.util.List; public class RepoData { private static final String TAG = "RepoData"; private final Object populateLock = new Object(); public final String url; public final String id; public final File cacheRoot; public final SharedPreferences cachedPreferences; public final File metaDataCache; public final HashMap moduleHashMap; public long lastUpdate; public String name; private boolean enabled; // Cache for speed protected RepoData(String url, File cacheRoot, SharedPreferences cachedPreferences) { this.url = url; this.id = RepoManager.internalIdOfUrl(url); this.cacheRoot = cacheRoot; this.cachedPreferences = cachedPreferences; this.metaDataCache = new File(cacheRoot, "modules.json"); this.moduleHashMap = new HashMap<>(); this.name = this.url; // Set url as default name this.enabled = MainApplication.getSharedPreferences() .getBoolean("pref_" + this.id + "_enabled", this.isEnabledByDefault(this.id)); if (!this.cacheRoot.isDirectory()) { this.cacheRoot.mkdirs(); } else { if (this.metaDataCache.exists()) { this.lastUpdate = metaDataCache.lastModified(); if (this.lastUpdate > System.currentTimeMillis()) { this.lastUpdate = 0; // Don't allow time travel } try { List modules = this.populate(new JSONObject( new String(Files.read(this.metaDataCache), StandardCharsets.UTF_8))); for (RepoModule repoModule : modules) { if (!this.tryLoadMetadata(repoModule)) { repoModule.moduleInfo.flags &= ~ModuleInfo.FLAG_METADATA_INVALID; } } } catch (Exception e) { this.metaDataCache.delete(); } } } } protected boolean prepare() { return true; } protected List populate(JSONObject jsonObject) throws JSONException { List newModules = new ArrayList<>(); synchronized (this.populateLock) { String name = jsonObject.getString("name").trim(); String nameForModules = name.endsWith(" (Official)") ? name.substring(0, name.length() - 11) : name; long lastUpdate = jsonObject.getLong("last_update"); for (RepoModule repoModule : this.moduleHashMap.values()) { repoModule.processed = false; } JSONArray array = jsonObject.getJSONArray("modules"); int len = array.length(); for (int i = 0; i < len; i++) { JSONObject module = array.getJSONObject(i); String moduleId = module.getString("id"); // Deny remote modules ids shorter than 3 chars if (moduleId.length() < 3) continue; long moduleLastUpdate = module.getLong("last_update"); String moduleNotesUrl = module.getString("notes_url"); String modulePropsUrl = module.getString("prop_url"); String moduleZipUrl = module.getString("zip_url"); String moduleChecksum = module.optString("checksum"); String moduleStars = module.optString("stars"); RepoModule repoModule = this.moduleHashMap.get(moduleId); if (repoModule == null) { repoModule = new RepoModule(this, moduleId); this.moduleHashMap.put(moduleId, repoModule); newModules.add(repoModule); } else { if (repoModule.lastUpdated < moduleLastUpdate || repoModule.moduleInfo.hasFlag(ModuleInfo.FLAG_METADATA_INVALID)) { newModules.add(repoModule); } } repoModule.processed = true; repoModule.repoName = nameForModules; repoModule.lastUpdated = moduleLastUpdate; repoModule.notesUrl = moduleNotesUrl; repoModule.propUrl = modulePropsUrl; repoModule.zipUrl = moduleZipUrl; repoModule.checksum = moduleChecksum; if (!moduleStars.isEmpty()) { try { repoModule.qualityValue = Integer.parseInt(moduleStars); repoModule.qualityText = R.string.module_stars; } catch (NumberFormatException ignored) {} } } // Remove no longer existing modules Iterator moduleInfoIterator = this.moduleHashMap.values().iterator(); while (moduleInfoIterator.hasNext()) { RepoModule repoModule = moduleInfoIterator.next(); if (!repoModule.processed) { new File(this.cacheRoot, repoModule.id + ".prop").delete(); moduleInfoIterator.remove(); } } // Update final metadata this.name = name; this.lastUpdate = lastUpdate; } return newModules; } protected boolean isEnabledByDefault(String id) { return !BuildConfig.DISABLED_REPOS.contains(id); } public void storeMetadata(RepoModule repoModule,byte[] data) throws IOException { Files.write(new File(this.cacheRoot, repoModule.id + ".prop"), data); } public boolean tryLoadMetadata(RepoModule repoModule) { File file = new File(this.cacheRoot, repoModule.id + ".prop"); if (file.exists()) { try { ModuleInfo moduleInfo = repoModule.moduleInfo; PropUtils.readProperties(moduleInfo, file.getAbsolutePath(), repoModule.repoName + "/" + moduleInfo.name, false); moduleInfo.flags &= ~ModuleInfo.FLAG_METADATA_INVALID; if (moduleInfo.version == null) { moduleInfo.version = "v" + moduleInfo.versionCode; } return true; } catch (Exception ignored) { file.delete(); } } repoModule.moduleInfo.flags |= ModuleInfo.FLAG_METADATA_INVALID; return false; } public String getNameOrFallback(String fallback) { return this.name == null || this.name.equals(this.url) ? fallback : this.name; } public boolean isEnabled() { return this.enabled; } public void setEnabled(boolean enabled) { this.enabled = enabled; MainApplication.getSharedPreferences().edit() .putBoolean("pref_" + this.id + "_enabled", enabled).apply(); } public void updateEnabledState() { this.enabled = MainApplication.getSharedPreferences() .getBoolean("pref_" + this.id + "_enabled", this.isEnabledByDefault(this.id)); } }