Skip to content

๐Ÿ”ข Remotely manage your Flutter app's versioning and availability

License

Notifications You must be signed in to change notification settings

kamranbekirovyz/versionarte

Repository files navigation

versionarte

Remotely manage your Flutter app's versioning and availability, with a variety of helpful, and in some cases life-saving features with total freedom over the UI allowing you to customize the user experience to fit your app's branding and style.

cover_picture

Features can be implemented with versionarte:

  • โœ‹ Force users to update to the latest version of your app before continuing
  • ๐Ÿ’†๐Ÿปโ€โ™‚๏ธ Have separate minimum, latest versions and availability status for platforms
  • ๐Ÿšง Disable your app for maintenance with custom information text
  • ๐Ÿ†• Inform users when an optional update is available
  • ๐Ÿ”— Launch the App Store on iOS and Play Store on Android

๐Ÿ Getting started

How this works? Call Versionarte.check method by providing it a VersionarteProvider (an object responsible for fetching the versioning information from a remote service) which returns a VersionarteResult (an object containing app's versioning and availability information).

There are 2 built-in providers, RemoteConfigVersionarteProvider and RestfulVersionarteProvider, which fetches the versioning information from Firebase Remote Config and RESTful API respectively. You can also create your own custom provider by extending the VersionarteProvider class.

๐Ÿ’ก No need to try-catch the Versionarte.check method, as it catches all the errors internally and if something goes wrong, an instance of VersionarteResult with status VersionarteStatus.unknown is still returned.
๐Ÿ’ก Be sure to check the debug console to see insightful debug-only prints.

1. Using Firebase Remote Config

The RemoteConfigVersionarteProvider fetches information stored in Firebase Remote Config with the key name of "versionarte". You need to set up the Firebase Remote Config service before using this provider. See Firebase Remote Config setup guide to learn more about configuration.

Example:

final result = await Versionarte.check(
    versionarteProvider: RemoteConfigVersionarteProvider(),
);

Optional parameters:

  • keyName: key name for the Firebase Remote Config to fetch. By default, it's set to "versionarte". Specify if you upload the configuration JSON using a different key name.
  • initializeInternally: if your project already initializes and configures Firebase Remote Config, set this to false. By default, it's set to true.
  • remoteConfigSettings: settings for Firebase Remote Config if initializeInternally set to true. By default, fetchTimeout and minimumFetchInterval are set to 10 seconds.

2. Using RESTful API

The RestfulVersionarteProvider fetches versioning and availability information by sending HTTP GET request to the specified URL with optional headers. The response body should be a JSON string that follows the configuration JSON format.

Example:

final result = await Versionarte.check(
    versionarteProvider: RestfulVersionarteProvider(
        url: 'https://myapi.com/getVersioning',
    ),
);

Optional parameters:

  • headers: headers to send with the HTTP GET request. By default, it's set to an empty map.

3. Using custom VersionarteProvider

To use remote services to provide versioning and availability information of your app, extend the VersionarteProvider class and override the getStoreVersioning method which is responsible for fetching the information and returning it as a StoreVersioning object.

class MyCustomVersionarteProvider extends VersionarteProvider {
  @override
  Future<StoreVersioning> getStoreVersioning() async {
    final result = MyCustomService.fetchVersioning();

    final decodedResult = jsonDecode(result);

    return StoreVersioning.fromJson(decodedResult);
  }

Example:

final result = await Versionarte.check(
    versionarteProvider: MyCustomVersionarteProvider(),
);

๐Ÿ™Œ Handle the result

Obtained VersionarteResult has 3 parameters:

  • status: (VersionarteResult) the status of the app. It can be one of the following values:
    • VersionarteStatus.appInactive: the app is inactive for usage.
    • VersionarteStatus.mustUpdate: user must update before continuing.
    • VersionarteStatus.shouldUpdate: user can continue with and without updating.
    • VersionarteStatus.upToDate: the user's version is up to date.
    • VersionarteStatus.unknown: error occured while checking status.
  • details: (StorePlatformDetails) Details for the current platform, including messages for when the app is inactive.

Then, based on status do the if-else checks:

if (result.status == VersionarteResult.appInactive) {
    final message = result.details.status.getMessageForLanguage('en');
    // TODO: Handle the case where the app is inactive
} else if (result == VersionarteResult.mustUpdate) {
    // TODO: Handle the case where an update is required
} else if (result == VersionarteResult.shouldUpdate) {
    // TODO: Handle the case where an update is optional
} 

๐Ÿ”— Launching the stores

To launch the App Store on iOS and Play Store on Android, use the Versionarte.launchStore method by providing it with appleAppId (int) for App Store and androidPackageName (String) for Play Store.

Versionarte.launchStore(
    appleAppId: 123456789,
    androidPackageName: 'com.example.app',
);

๐Ÿ’ก androidPackageName is optional: if not provided corresponding value obtained from package_info will be used. ๐Ÿ’ก It's suggested to test launching store on a real device.

See the example directory for a complete sample app.

๐Ÿ–‹๏ธ Configuration JSON

For providing app's status and availability information, versionarte requires a specific JSON format. Whether you're using RemoteConfigVersionarteProvider, RestfulVersionarteProvider, or a custom VersionarteProvider, make sure to use this JSON format.

๐Ÿ’ก Information for all platforms in the JSON is not necessary: you can provide information for only one platform, or for two platforms, or for all three platforms.
๐Ÿ’ก While the app status is active, the message can be left empty or set to null.

{
    "android": {
        "version": {
            "minimum": "2.7.0",
            "latest": "2.8.0"
        },
        "status": {
            "active": true,
            "message": {
                "en": "App is in maintanence mode, please come back later.",
                "es": "La aplicaciรณn estรก en modo de mantenimiento, vuelva mรกs tarde."
            }
        }
    },
    "iOS": {
        // same data we used for "android"
    }
    "macOS": {
        // same data we used for "android"
    }
}

This JSON represents information stored separately for three platforms, containing the minimum and latest versions, and the availability status.

Each platform contains two objects:

  • version:
    • minimum: The minimum version of the app users can use.
    • latest: The latest version of the app available.
  • status:
    • active: A boolean that indicates whether the app is currently active or not.
    • message: A Map that contains the messages for different languages to be displayed to the user when app is inactive. The keys of the map represent the language codes (e.g., "en" for English, "es" for Spanish), and the values represent the message in that language.

๐Ÿž Bugs/Requests

If you encounter any problems please open an issue. If you feel the library is missing a feature, please raise a ticket on GitHub and we'll look into it. Pull requests are welcome.

๐Ÿ“ƒ License

MIT License