Vault


Source link: https://github.com/contentful/vault

Vault

Vault is an Android library that simplifies persistence of data from Contentful via SQLite. It lets you define Java representations for your Contentful models. Then, at compile-time it will create a corresponding database schema by generating all the required boilerplate code and injecting it to the classpath. It is also bundled with a complementary lightweight runtime which exposes a simple ORM-like API for pulling resources from the generated database.

Setup

Grab via Maven:

<dependency>
<groupId>com.contentful.vault</groupId>
<artifactId>compiler</artifactId>
<version>3.0.1/version> </dependency> <dependency>
<groupId>com.contentful.vault</groupId>
<artifactId>core</artifactId>
<version>3.0.1</version> </dependency>

or Gradle:

apt 'com.contentful.vault:compiler:3.0.1' compile 'com.contentful.vault:core:3.0.1'

or Gradle 3.+:

annotationProcessor 'com.contentful.vault:compiler:3.0.1' annotationProcessor 'com.contentful.vault:core:3.0.1' compile 'com.contentful.vault:core:3.0.1'

Note for Gradle users, make sure to use the android-apt Gradle plugin, which lets you configure compile-time only dependencies. Note for Gradle 3.0 users, please use the annotationProcessor instead of apt.

Snapshots of the development version are available in Sonatype's snapshots repository.

Models & Fields

Models are defined by declaring a subclass of the Resource class. Annotate the class with @ContentType, which takes the Content Type's ID as its value.

Fields are defined by annotating class attributes with the @Field annotation.

@ContentType("cat") public class Cat extends Resource {

@Field public String name;
@Field public Cat bestFriend;
@Field public Asset image; 
}

By default, the name of the attribute will be used as the field's ID, but can also be specified explicitly:

@Field("field-id-goes-here")  public String someField; 

Field ids are escaped and so can be used regularly, however when making queries with a WHERE condition it is up to the caller to escape the field name in case it is a reserved keyword. For example, consider the following model:

@ContentType("...") public class Foo extends Resource {

@Field public String order; 
}

Since order is a reserved SQLite keyword, in order to make a query which references that field here is how to escape it:

vault.fetch(Foo.class)
  .where("`" + Foo$Fields.ORDER "` = ?", "bar")
  .first();

Spaces

Spaces can be defined by declaring a class annotated with the @Space annotation. It is required to specify the Space ID, an array of model classes and an array of locale codes you wish to persist:

@Space(
  value = "cfexampleapi",
  models = {
 Cat.class 
}
,
  locales = {
 "en-US", "tlh" 
}
 ) public class DemoSpace {
 
}

Synchronization

Once a Space is defined, invoke Vault to sync the local database with the remote state:

// Client CDAClient client = CDAClient.builder()
  .setSpace("cfexampleapi")
  .setToken("b4c0n73n7fu1")
  .build();
  // Sync Vault.with(context, DemoSpace.class).requestSync(client);

Vault will use a worker thread to request incremental updates from the Sync API and reflect the remote changes to its database. Once sync is completed, Vault will fire a broadcast with the action Vault.ACTION_SYNC_COMPLETE.

Alternatively, you can provide a SyncCallback which will be invoked once sync is completed, but make sure to cancel it according to its host lifecycle events:

class SomeActivity extends Activity {

SyncCallback callback;

@Override protected void onCreate(Bundle savedInstanceState) {

  super.onCreate(savedInstanceState);

 Vault.with(this, DemoSpace.class).requestSync(client, callback = new SyncCallback() {

 @Override public void onResult(SyncResult result) {

if (result.isSuccessful()) {

  // Success \o/

}
 else {

  // Failure

}

 
}

  
}
);

}

@Override protected void onDestroy() {

  Vault.cancel(callback);

  super.onDestroy();

}
 
}

Similarly, if you're using RxJava, you can be notified of sync results via an Observable:

Vault.observeSyncResults() // returns Observable<SyncResult>

Queries

Vault provides a wrapper around its generated database which can be easily used to fetch persisted objects, some examples:

Vault vault = Vault.with(context, DemoSpace.class);
  // Fetch the first Cat vault.fetch(Cat.class)
  .first();

// Fetch the most recently updated Cat vault.fetch(Cat.class)
  .order(Cat$Fields.UPDATED_AT + " DESC")
  .first();
  // Fetch a Cat with a specific name vault.fetch(Cat.class)
  .where(Cat$Fields.NAME + " = ?", "Nyan Cat")
  .first();
  // Fetch a Cat with a specific name pattern vault.fetch(Cat.class)
  .where(Cat$Fields.NAME + " LIKE ?", "%Nyan%")
  .first();
  // Fetch a Cat with a specific boolean field // SQLite is storing booleans as 0/1  vault.fetch(Cat.class)
  .where(Cat$Fields.IS_GRUMPY + " = ?", "1")
  .first();

// Fetch all Cats, ordered by creation date: vault.fetch(Cat.class)
  .order(Cat$Fields.CREATED_AT)
  .all();
  // Fetch all Cats, using the Klingon locale: vault.fetch(Cat.class)
  .all("tlh");

If you're using RxJava, you can create queries with the observe() method, for example:

vault.observe(Cat.class)
  .where(Cat$Fields.NAME + " = ?", "Happy Cat")
  .all() // returns Observable<Cat>

The above example creates an Observable that subscribes and observes on the same thread initiating the query, so make sure to change that according to your requirements. A typical use-case would be:

vault.observe(Cat.class)
  .all()
  .subscribeOn(Schedulers.io())
  .observeOn(AndroidSchedulers.mainThread())

Migrations

Whenever changes are introduced to any of the previously used models, a migration has to be applied. Simply increment the version number for your space to trigger a migration:

@Space(value = "cfexampleapi", models = {
 Cat.class 
}
, dbVersion = 2) public class DemoSpace {
 
}

Note: this will delete any previously persisted data.

Database pre-seeding

Depending on the amount of content in a given space, initial synchronization might take some time. For that we've added support to pre-seed the database with static content. In order to do that one has to add an existing Vault database file into the assets folder of the project, and then reference this file through the copyPath attribute, for example:

@Space(value = "foo", models = {
 Bar.class 
}
, copyPath = "vault_file_name.db") public class FooSpace {
 
}

Note that in order to add this functionality to an already shipped app, the dbVersion value has to be increased, as it causes invalidation of any pre-existing content.

ProGuard

Grab the ProGuard configuration file and apply to your project.

Documentation

Javadoc is available here.

License

Copyright 2017 Contentful, GmbH.  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. 

Resources

A delivery service for Android Wear. Courier uses Wearable.DataApi and Wearable.MessageApi to deliver objects between devices simply and cleanly.

Unofficial ThingSpeak API library for Android.

Features:

Android SQLite database manager.

Truth is a testing framework designed to make your tests and their error messages more readable and discoverable, while being extensible to new types of objects.

Truth adopts a fluent style for test propositions, is extensible in several ways, supports IDE completion/discovery of available propositions, and supports different responses to un-true propositions. Truth can be used to declare JUnit-style assumptions (which skip the test on failure), assertions (interrupt the test on failure), and expectations (continue the test, but collect errors and report failure at the end).

Use gradle tasks to run specific adb commands. You can use this plugin to do things such as:

  • Find all devices attached and get basic info about them
  • Select the first one that complies with a custom rule
  • Install a specific APK from the available build types + flavours
  • Clear preferences or do something related to the APK to prepare for tests
  • Run monkey for that specific APK on that specific device
  • Uninstall the APK

Bash script that exports content provider's data to raw and CSV format. It can be your own content provider or any other provider that android decided not to block with permissions.

Topics


2D Engines   3D Engines   9-Patch   Action Bars   Activities   ADB   Advertisements   Analytics   Animations   ANR   AOP   API   APK   APT   Architecture   Audio   Autocomplete   Background Processing   Backward Compatibility   Badges   Bar Codes   Benchmarking   Bitmaps   Bluetooth   Blur Effects   Bread Crumbs   BRMS   Browser Extensions   Build Systems   Bundles   Buttons   Caching   Camera   Canvas   Cards   Carousels   Changelog   Checkboxes   Cloud Storages   Color Analysis   Color Pickers   Colors   Comet/Push   Compass Sensors   Conferences   Content Providers   Continuous Integration   Crash Reports   Credit Cards   Credits   CSV   Curl/Flip   Data Binding   Data Generators   Data Structures   Database   Database Browsers   Date &   Debugging   Decompilers   Deep Links   Dependency Injections   Design   Design Patterns   Dex   Dialogs   Distributed Computing   Distribution Platforms   Download Managers   Drawables   Emoji   Emulators   EPUB   Equalizers &   Event Buses   Exception Handling   Face Recognition   Feedback &   File System   File/Directory   Fingerprint   Floating Action   Fonts   Forms   Fragments   FRP   FSM   Functional Programming   Gamepads   Games   Geocaching   Gestures   GIF   Glow Pad   Gradle Plugins   Graphics   Grid Views   Highlighting   HTML   HTTP Mocking   Icons   IDE   IDE Plugins   Image Croppers   Image Loaders   Image Pickers   Image Processing   Image Views   Instrumentation   Intents   Job Schedulers   JSON   Keyboard   Kotlin   Layouts   Library Demos   List View   List Views   Localization   Location   Lock Patterns   Logcat   Logging   Mails   Maps   Markdown   Mathematics   Maven Plugins   MBaaS   Media   Menus   Messaging   MIME   Mobile Web   Native Image   Navigation   NDK   Networking   NFC   NoSQL   Number Pickers   OAuth   Object Mocking   OCR Engines   OpenGL   ORM   Other Pickers   Parallax List   Parcelables   Particle Systems   Password Inputs   PDF   Permissions   Physics Engines   Platforms   Plugin Frameworks   Preferences   Progress Indicators   ProGuard   Properties   Protocol Buffer   Pull To   Purchases   Push/Pull   QR Codes   Quick Return   Radio Buttons   Range Bars   Ratings   Recycler Views   Resources   REST   Ripple Effects   RSS   Screenshots   Scripting   Scroll Views   SDK   Search Inputs   Security   Sensors   Services   Showcase Views   Signatures   Sliding Panels   Snackbars   SOAP   Social Networks   Spannable   Spinners   Splash Screens   SSH   Static Analysis   Status Bars   Styling   SVG   System   Tags   Task Managers   TDD &   Template Engines   Testing   Testing Tools   Text Formatting   Text Views   Text Watchers   Text-to   Toasts   Toolkits For   Tools   Tooltips   Trainings   TV   Twitter   Updaters   USB   User Stories   Utils   Validation   Video   View Adapters   View Pagers   Views   Watch Face   Wearable Data   Wearables   Weather   Web Tools   Web Views   WebRTC   WebSockets   Wheel Widgets   Wi-Fi   Widgets   Windows   Wizards   XML   XMPP   YAML   ZIP Codes