/*
* Copyright (C) 2014 Simon Vig Therkildsen
*
* 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 net.simonvt.cathode.remote.sync;
import android.content.ContentProviderOperation;
import android.content.ContentResolver;
import android.content.OperationApplicationException;
import android.database.Cursor;
import android.os.RemoteException;
import java.util.ArrayList;
import java.util.List;
import javax.inject.Inject;
import net.simonvt.cathode.BuildConfig;
import net.simonvt.cathode.api.entity.Watching;
import net.simonvt.cathode.api.enumeration.Action;
import net.simonvt.cathode.api.service.UsersService;
import net.simonvt.cathode.jobqueue.JobFailedException;
import net.simonvt.cathode.provider.DatabaseContract.EpisodeColumns;
import net.simonvt.cathode.provider.DatabaseContract.MovieColumns;
import net.simonvt.cathode.provider.DatabaseSchematic.Tables;
import net.simonvt.cathode.provider.EpisodeDatabaseHelper;
import net.simonvt.cathode.provider.MovieDatabaseHelper;
import net.simonvt.cathode.provider.ProviderSchematic.Episodes;
import net.simonvt.cathode.provider.ProviderSchematic.Movies;
import net.simonvt.cathode.provider.SeasonDatabaseHelper;
import net.simonvt.cathode.provider.ShowDatabaseHelper;
import net.simonvt.cathode.remote.CallJob;
import net.simonvt.cathode.remote.Flags;
import net.simonvt.cathode.remote.sync.movies.SyncMovie;
import net.simonvt.cathode.remote.sync.movies.SyncWatchedMovies;
import net.simonvt.cathode.remote.sync.shows.SyncSeason;
import net.simonvt.cathode.remote.sync.shows.SyncShow;
import net.simonvt.cathode.remote.sync.shows.SyncShowWatchedStatus;
import net.simonvt.schematic.Cursors;
import retrofit2.Call;
import timber.log.Timber;
public class SyncWatching extends CallJob<Watching> {
@Inject transient UsersService usersService;
@Inject transient ShowDatabaseHelper showHelper;
@Inject transient SeasonDatabaseHelper seasonHelper;
@Inject transient EpisodeDatabaseHelper episodeHelper;
@Inject transient MovieDatabaseHelper movieHelper;
public SyncWatching() {
super(Flags.REQUIRES_AUTH);
}
@Override public String key() {
return "SyncWatching";
}
@Override public int getPriority() {
return PRIORITY_ACTIONS;
}
@Override public Call<Watching> getCall() {
return usersService.watching();
}
@Override public void handleResponse(Watching watching) {
ContentResolver resolver = getContentResolver();
ArrayList<ContentProviderOperation> ops = new ArrayList<>();
Cursor episodeWatchingCursor =
getContentResolver().query(Episodes.EPISODE_WATCHING, new String[] {
Tables.EPISODES + "." + EpisodeColumns.ID,
}, null, null, null);
List<Long> episodeWatching = new ArrayList<>();
while (episodeWatchingCursor.moveToNext()) {
episodeWatching.add(Cursors.getLong(episodeWatchingCursor, EpisodeColumns.ID));
}
episodeWatchingCursor.close();
Cursor movieWatchingCursor = getContentResolver().query(Movies.WATCHING, new String[] {
MovieColumns.ID,
}, null, null, null);
List<Long> movieWatching = new ArrayList<>();
while (movieWatchingCursor.moveToNext()) {
movieWatching.add(Cursors.getLong(movieWatchingCursor, MovieColumns.ID));
}
movieWatchingCursor.close();
ContentProviderOperation op = null;
if (watching != null) {
if (watching.getType() != null) {
switch (watching.getType()) {
case EPISODE: {
final long showTraktId = watching.getShow().getIds().getTrakt();
ShowDatabaseHelper.IdResult showResult = showHelper.getIdOrCreate(showTraktId);
final long showId = showResult.showId;
if (showHelper.needsSync(showId)) {
queue(new SyncShow(showTraktId));
}
final boolean didShowExist = !showResult.didCreate;
final int seasonNumber = watching.getEpisode().getSeason();
SeasonDatabaseHelper.IdResult seasonResult =
seasonHelper.getIdOrCreate(showId, seasonNumber);
final long seasonId = seasonResult.id;
final boolean didSeasonExist = !seasonResult.didCreate;
if (seasonResult.didCreate) {
if (didShowExist) {
queue(new SyncShow(showTraktId));
}
}
final int episodeNumber = watching.getEpisode().getNumber();
EpisodeDatabaseHelper.IdResult episodeResult =
episodeHelper.getIdOrCreate(showId, seasonId, episodeNumber);
final long episodeId = episodeResult.id;
if (episodeResult.didCreate) {
if (didShowExist && didSeasonExist) {
queue(new SyncSeason(showTraktId, seasonNumber));
}
}
episodeWatching.remove(episodeId);
if (watching.getAction() == Action.CHECKIN) {
op = ContentProviderOperation.newUpdate(Episodes.withId(episodeId))
.withValue(EpisodeColumns.CHECKED_IN, true)
.withValue(EpisodeColumns.WATCHING, false)
.withValue(EpisodeColumns.STARTED_AT, watching.getStartedAt().getTimeInMillis())
.withValue(EpisodeColumns.EXPIRES_AT, watching.getExpiresAt().getTimeInMillis())
.build();
} else {
op = ContentProviderOperation.newUpdate(Episodes.withId(episodeId))
.withValue(EpisodeColumns.CHECKED_IN, false)
.withValue(EpisodeColumns.WATCHING, true)
.withValue(EpisodeColumns.STARTED_AT, watching.getStartedAt().getTimeInMillis())
.withValue(EpisodeColumns.EXPIRES_AT, watching.getExpiresAt().getTimeInMillis())
.build();
}
ops.add(op);
break;
}
case MOVIE: {
final long movieTraktId = watching.getMovie().getIds().getTrakt();
MovieDatabaseHelper.IdResult result = movieHelper.getIdOrCreate(movieTraktId);
final long movieId = result.movieId;
if (movieHelper.needsSync(movieId)) {
queue(new SyncMovie(movieTraktId));
}
movieWatching.remove(movieId);
if (watching.getAction() == Action.CHECKIN) {
op = ContentProviderOperation.newUpdate(Movies.withId(movieId))
.withValue(MovieColumns.CHECKED_IN, true)
.withValue(MovieColumns.WATCHING, false)
.withValue(MovieColumns.STARTED_AT, watching.getStartedAt().getTimeInMillis())
.withValue(MovieColumns.EXPIRES_AT, watching.getExpiresAt().getTimeInMillis())
.build();
} else {
op = ContentProviderOperation.newUpdate(Movies.withId(movieId))
.withValue(MovieColumns.CHECKED_IN, false)
.withValue(MovieColumns.WATCHING, true)
.withValue(MovieColumns.STARTED_AT, watching.getStartedAt().getTimeInMillis())
.withValue(MovieColumns.EXPIRES_AT, watching.getExpiresAt().getTimeInMillis())
.build();
}
ops.add(op);
break;
}
}
}
}
for (Long episodeId : episodeWatching) {
final long showId = episodeHelper.getShowId(episodeId);
final long showTraktId = showHelper.getTraktId(showId);
queue(new SyncShowWatchedStatus(showTraktId));
op = ContentProviderOperation.newUpdate(Episodes.withId(episodeId))
.withValue(EpisodeColumns.CHECKED_IN, false)
.withValue(EpisodeColumns.WATCHING, false)
.build();
ops.add(op);
}
for (Long movieId : movieWatching) {
queue(new SyncWatchedMovies());
op = ContentProviderOperation.newUpdate(Movies.withId(movieId))
.withValue(MovieColumns.CHECKED_IN, false)
.withValue(MovieColumns.WATCHING, false)
.build();
ops.add(op);
}
try {
resolver.applyBatch(BuildConfig.PROVIDER_AUTHORITY, ops);
} catch (RemoteException e) {
Timber.e(e, "SyncWatchingTask failed");
throw new JobFailedException(e);
} catch (OperationApplicationException e) {
Timber.e(e, "SyncWatchingTask failed");
throw new JobFailedException(e);
}
}
}