forked from mikehardy/react-native-update-apk
-
Notifications
You must be signed in to change notification settings - Fork 0
/
App.js
245 lines (217 loc) · 9.58 KB
/
App.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
/**
* Sample React Native App
* https://github.com/facebook/react-native
*
* @format
* @flow
* @lint-ignore-every XPLATJSCOPYRIGHT1
*/
import React, { Component } from "react";
import {
Alert,
Button,
SafeAreaView,
ScrollView,
StyleSheet,
Text
} from "react-native";
import * as UpdateAPK from "rn-update-apk";
type Props = {};
export default class App extends Component<Props> {
constructor(props) {
super(props);
this.state = {
// If you have something in state, you will be able to provide status to users
downloadProgress: -1,
allApps: [],
allNonSystemApps: [],
};
updater = new UpdateAPK.UpdateAPK({
// iOS must use App Store and this is the app ID. This is a sample: "All Birds of Ecuador" (¡Qué lindo!)
iosAppId: "1104809018",
apkVersionUrl:
"https://raw.githubusercontent.com/mikehardy/react-native-update-apk/master/example/test-version.json",
//apkVersionOptions is optional, you should use it if you need to pass options to fetch request
apkVersionOptions: {
method:'GET',
headers: {}
},
//apkOptions is optional
//Complements or replaces the DownloadFileOptions (from react-native-fs) to download the new APK
//By default the following options are already set: fromUrl, toFile, begin, progress, background and progressDivider
//You should use it if you need to pass additional information (for example: headers) to download the new APK
apkOptions: {
headers: {}
},
// The name of this 'fileProviderAuthority' is defined in AndroidManifest.xml. THEY MUST MATCH.
// By default other modules like rn-fetch-blob define one (conveniently named the same as below)
// but if you don't match the names you will get an odd-looking XML exception:
// "Attempt to invoke virtual method 'android.content.res.XmlResourceParser ....' on a null object reference"
fileProviderAuthority: "com.example.provider",
// This callback is called if there is a new version but it is not a forceUpdate.
needUpdateApp: needUpdate => {
Alert.alert(
"Update Available",
"New version released, do you want to update? " +
"(TESTING NOTE 1: stop your dev package server now - or the test package will try to load from it " +
"instead of the included bundle leading to Javascript/Native incompatibilities." +
"TESTING NOTE 2: the version is fixed at 1.0 so example test updates always work. " +
"Compare the Last Update Times to verify it installed)",
[
{ text: "Cancel", onPress: () => {} },
// Note, apps can be large. You may want to check if the network is metered (cellular data) to be nice.
// Note that the user will likely get a popup saying the device is set to block installs from uknown sources.
// ...you will need to guide them through that, maybe by explaining it here, before you call needUpdate(true);
{ text: "Update", onPress: () => needUpdate(true) }
]
);
},
// This will be called before the download/update where you defined forceUpdate: true in the version JSON
forceUpdateApp: () => {
console.log("forceUpdateApp callback called");
},
// Called if the current version appears to be the most recent available
notNeedUpdateApp: () => {
console.log("notNeedUpdateApp callback called");
},
// This is passed to react-native-fs as a callback
downloadApkStart: () => {
console.log("downloadApkStart callback called");
},
// Called with 0-99 for progress during the download
downloadApkProgress: progress => {
console.log(`downloadApkProgress callback called - ${progress}%...`);
// This is your opportunity to provide feedback to users on download progress
// If you hae a state variable it is trivial to update the UI
this.setState({ downloadProgress: progress });
},
// This is called prior to the update. If you throw it will abort the update
downloadApkEnd: () => {
// This could be an opportunity to check the APK signature thumbprints,
// If they mismatch your update will fail, the user will have to uninstall first.
// If you implement SHAsums on the file you could detect tampering here as well
// Finally for APK25+ you should check REQUEST_INSTALL_PACKAGES permission
// prior to the attempt at some point, and provide guidance about "unknown sources" etc
console.log("downloadApkEnd callback called");
},
// This is called if the fetch of the version or the APK fails, so should be generic
onError: err => {
console.log("onError callback called", err);
Alert.alert("There was an error", err.message);
}
});
}
async componentDidMount() {
// If you want to update devices below Android 5, they have SSL issues with some servers.
// You will get a protocol error unless you patch the SSL Provider.
// This will fail if they don't have Google Play Services installed though.
// You can optionally force the patch on Android 5+ with boolean param 1
// You can also optionally display a Google dialog for user repair (if possible) with boolean param 2
UpdateAPK.patchSSLProvider()
.then(ret => {
// This means
console.log("SSL Provider Patch proceeded without error");
})
.catch(rej => {
// Modern SSL servers have gotten more strict about which SSL/TLS protocols/versions they allow.
// If you arrived here, you have an old device with an unpatchable SSL Provider, your downloads will probably fail.
// You should provide some sort of messaging to these users or provide updates over HTTP as needed
// Luckily this only applies to Android 4.x without Google Play Services, a very small percentage.
console.log("SSL Provider patch failed", rej);
let message = "Old Android API, and SSL Provider could not be patched.";
if (rej.message.includes("repairable")) {
// In this particular case the user may even be able to fix it with a Google Play Services update
message +=
" This is repairable on this device though." +
" You should send the users to the Play Store to update Play Services...";
Alert.alert("Possible SSL Problem", message);
UpdateAPK.patchSSLProvider(false, true); // This will ask Google Play Services to help the user repair
} else {
Alert.alert("Possible SSL Problem", message);
}
});
UpdateAPK.getApps().then(apps => {
console.log("Installed Apps: ", JSON.stringify(apps));
this.setState({ allApps: apps});
}).catch(e => console.log("Unable to getApps?", e));
UpdateAPK.getNonSystemApps().then(apps => {
console.log("Installed Non-System Apps: ", JSON.stringify(apps));
this.setState({ allNonSystemApps: apps});
}).catch(e => console.log("Unable to getNonSystemApps?", e));
}
_onCheckServerVersion = () => {
console.log("checking for update");
updater.checkUpdate();
};
render() {
return (
<SafeAreaView style={styles.container}>
<Text style={styles.welcome}>rn-update-apk example</Text>
<Text style={styles.instructions}>
Installed Package Name: {UpdateAPK.getInstalledPackageName()}
</Text>
<Text style={styles.instructions}>
Installed Version Code: {UpdateAPK.getInstalledVersionCode()}
</Text>
<Text style={styles.instructions}>
Installed Version Name: {UpdateAPK.getInstalledVersionName()}
</Text>
<Text style={styles.instructions}>
Installed First Install Time:
{new Date(+UpdateAPK.getInstalledFirstInstallTime()).toUTCString()}
</Text>
<Text style={styles.instructions}>
Installed Last Update Time:
{new Date(+UpdateAPK.getInstalledLastUpdateTime()).toUTCString()}
</Text>
<Text style={styles.instructions}>
Installed Package Installer:
{UpdateAPK.getInstalledPackageInstaller()}
</Text>
<ScrollView style={{ flex: 1 }}>
<Text style={styles.instructions}>
Installed Apps: {JSON.stringify(this.state.allApps, null, '\t')}
</Text>
<Text style={styles.instructions}>
Installed Non-System Apps: {JSON.stringify(this.state.allNonSystemApps, null, '\t')}
</Text>
<Text style={styles.instructions}>
Installed Package Certificate SHA-256 Digest:
{ UpdateAPK.getInstalledSigningInfo() ? UpdateAPK.getInstalledSigningInfo()[0].thumbprint : "" }
</Text>
<Text style={styles.instructions}>
{ UpdateAPK.getInstalledSigningInfo() ? UpdateAPK.getInstalledSigningInfo()[0].toString : "" }
</Text>
</ScrollView>
{this.state.downloadProgress != -1 && (
<Text style={styles.instructions}>
Download Progress: {this.state.downloadProgress}%
</Text>
)}
<Button
title="Check Server For Update"
onPress={this._onCheckServerVersion}
/>
</SafeAreaView>
);
}
}
const styles = StyleSheet.create({
container: {
flex: 1,
justifyContent: "center",
alignItems: "center",
backgroundColor: "#F5FCFF"
},
welcome: {
fontSize: 20,
textAlign: "center",
margin: 10
},
instructions: {
fontSize: 12,
textAlign: "left",
color: "#333333",
marginBottom: 5
}
});