/* * Copyright (C) 2023 The Android Open Source Project * * 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 android.health.connect.aidl; import static android.health.connect.Constants.DEFAULT_INT; import static android.health.connect.Constants.DEFAULT_LONG; import android.annotation.NonNull; import android.annotation.Nullable; import android.health.connect.AggregateRecordsGroupedByDurationResponse; import android.health.connect.AggregateRecordsGroupedByPeriodResponse; import android.health.connect.AggregateRecordsResponse; import android.health.connect.AggregateResult; import android.health.connect.LocalTimeRangeFilter; import android.health.connect.TimeInstantRangeFilter; import android.health.connect.TimeRangeFilter; import android.health.connect.TimeRangeFilterHelper; import android.health.connect.datatypes.DataOrigin; import android.health.connect.internal.datatypes.utils.AggregationTypeIdMapper; import android.os.Parcel; import android.os.Parcelable; import android.util.ArrayMap; import java.time.Duration; import java.time.Instant; import java.time.LocalDateTime; import java.time.Period; import java.time.ZoneOffset; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.Objects; import java.util.Set; /** @hide */ public class AggregateDataResponseParcel implements Parcelable { public static final Creator CREATOR = new Creator<>() { @Override public AggregateDataResponseParcel createFromParcel(Parcel in) { return new AggregateDataResponseParcel(in); } @Override public AggregateDataResponseParcel[] newArray(int size) { return new AggregateDataResponseParcel[size]; } }; private final List> mAggregateRecordsResponses; private Duration mDuration; private Period mPeriod; private TimeRangeFilter mTimeRangeFilter; public AggregateDataResponseParcel(List> aggregateRecordsResponse) { mAggregateRecordsResponses = aggregateRecordsResponse; } protected AggregateDataResponseParcel(Parcel in) { final int size = in.readInt(); mAggregateRecordsResponses = new ArrayList<>(size); for (int i = 0; i < size; i++) { final int mapSize = in.readInt(); Map> result = new ArrayMap<>(mapSize); for (int mapI = 0; mapI < mapSize; mapI++) { int id = in.readInt(); boolean hasValue = in.readBoolean(); if (hasValue) { result.put( id, AggregationTypeIdMapper.getInstance() .getAggregateResultFor(id, in) .setZoneOffset(parseZoneOffset(in)) .setDataOrigins(in.createStringArrayList())); } else { result.put(id, null); } } mAggregateRecordsResponses.add(new AggregateRecordsResponse<>(result)); } int period = in.readInt(); if (period != DEFAULT_INT) { mPeriod = Period.ofDays(period); } long duration = in.readLong(); if (duration != DEFAULT_LONG) { mDuration = Duration.ofMillis(duration); } boolean isLocaltimeFilter = in.readBoolean(); long startTime = in.readLong(); long endTime = in.readLong(); if (startTime != DEFAULT_LONG && endTime != DEFAULT_LONG) { if (isLocaltimeFilter) { mTimeRangeFilter = new LocalTimeRangeFilter.Builder() .setStartTime( TimeRangeFilterHelper.getLocalTimeFromMillis(startTime)) .setEndTime(TimeRangeFilterHelper.getLocalTimeFromMillis(endTime)) .build(); } else { mTimeRangeFilter = new TimeInstantRangeFilter.Builder() .setStartTime(Instant.ofEpochMilli(startTime)) .setEndTime(Instant.ofEpochMilli(endTime)) .build(); } } } public AggregateDataResponseParcel setDuration( @Nullable Duration duration, @Nullable TimeRangeFilter timeRangeFilter) { mDuration = duration; mTimeRangeFilter = timeRangeFilter; return this; } public AggregateDataResponseParcel setPeriod( @Nullable Period period, @Nullable TimeRangeFilter timeRangeFilter) { mPeriod = period; mTimeRangeFilter = timeRangeFilter; return this; } /** * @return the first response from {@code mAggregateRecordsResponses} */ public AggregateRecordsResponse getAggregateDataResponse() { return mAggregateRecordsResponses.get(0); } /** * @return responses from {@code mAggregateRecordsResponses} grouped as per the {@code * mDuration} */ public List> getAggregateDataResponseGroupedByDuration() { Objects.requireNonNull(mDuration); List> aggregateRecordsGroupedByDurationResponse = new ArrayList<>(); long mStartTime = TimeRangeFilterHelper.getFilterStartTimeMillis(mTimeRangeFilter); long mEndTime = TimeRangeFilterHelper.getFilterEndTimeMillis(mTimeRangeFilter); long mDelta = getDurationDelta(mDuration); for (AggregateRecordsResponse aggregateRecordsResponse : mAggregateRecordsResponses) { aggregateRecordsGroupedByDurationResponse.add( new AggregateRecordsGroupedByDurationResponse<>( getDurationInstant(mStartTime), getDurationInstant(Math.min(mStartTime + mDelta, mEndTime)), aggregateRecordsResponse.getAggregateResults())); mStartTime += mDelta; } return aggregateRecordsGroupedByDurationResponse; } /** * @return responses from {@code mAggregateRecordsResponses} grouped as per the {@code mPeriod} */ public List> getAggregateDataResponseGroupedByPeriod() { Objects.requireNonNull(mPeriod); List> aggregateRecordsGroupedByPeriodResponses = new ArrayList<>(); LocalDateTime groupBoundary = ((LocalTimeRangeFilter) mTimeRangeFilter).getStartTime(); long mDelta = getPeriodDeltaInDays(mPeriod); for (AggregateRecordsResponse aggregateRecordsResponse : mAggregateRecordsResponses) { aggregateRecordsGroupedByPeriodResponses.add( new AggregateRecordsGroupedByPeriodResponse<>( groupBoundary, groupBoundary.plusDays(mDelta), aggregateRecordsResponse.getAggregateResults())); groupBoundary = groupBoundary.plusDays(mDelta); } if (!aggregateRecordsGroupedByPeriodResponses.isEmpty()) { aggregateRecordsGroupedByPeriodResponses .get(aggregateRecordsGroupedByPeriodResponses.size() - 1) .setEndTime(getPeriodEndLocalDateTime(mTimeRangeFilter)); } return aggregateRecordsGroupedByPeriodResponses; } @Override public int describeContents() { return 0; } @Override public void writeToParcel(@NonNull Parcel dest, int flags) { dest.writeInt(mAggregateRecordsResponses.size()); for (AggregateRecordsResponse aggregateRecordsResponse : mAggregateRecordsResponses) { dest.writeInt(aggregateRecordsResponse.getAggregateResults().size()); aggregateRecordsResponse .getAggregateResults() .forEach( (key, val) -> { dest.writeInt(key); // to represent if the value is present or not dest.writeBoolean(val != null); if (val != null) { val.putToParcel(dest); ZoneOffset zoneOffset = val.getZoneOffset(); if (zoneOffset != null) { dest.writeInt(val.getZoneOffset().getTotalSeconds()); } else { dest.writeInt(DEFAULT_INT); } Set dataOrigins = val.getDataOrigins(); List packageNames = new ArrayList<>(); for (DataOrigin dataOrigin : dataOrigins) { packageNames.add(dataOrigin.getPackageName()); } dest.writeStringList(packageNames); } }); } if (mPeriod != null) { dest.writeInt(mPeriod.getDays()); } else { dest.writeInt(DEFAULT_INT); } if (mDuration != null) { dest.writeLong(mDuration.toMillis()); } else { dest.writeLong(DEFAULT_LONG); } if (mTimeRangeFilter != null) { dest.writeBoolean(TimeRangeFilterHelper.isLocalTimeFilter(mTimeRangeFilter)); dest.writeLong(TimeRangeFilterHelper.getFilterStartTimeMillis(mTimeRangeFilter)); dest.writeLong(TimeRangeFilterHelper.getFilterEndTimeMillis(mTimeRangeFilter)); } else { dest.writeBoolean(false); dest.writeLong(DEFAULT_LONG); dest.writeLong(DEFAULT_LONG); } } private ZoneOffset parseZoneOffset(Parcel in) { int zoneOffsetInSecs = in.readInt(); ZoneOffset zoneOffset = null; if (zoneOffsetInSecs != DEFAULT_INT) { zoneOffset = ZoneOffset.ofTotalSeconds(zoneOffsetInSecs); } return zoneOffset; } private LocalDateTime getPeriodEndLocalDateTime(TimeRangeFilter timeRangeFilter) { if (timeRangeFilter instanceof TimeInstantRangeFilter) { return LocalDateTime.ofInstant( ((TimeInstantRangeFilter) timeRangeFilter).getEndTime(), ZoneOffset.systemDefault()); } else if (timeRangeFilter instanceof LocalTimeRangeFilter) { return ((LocalTimeRangeFilter) timeRangeFilter).getEndTime(); } else { throw new IllegalArgumentException( "Invalid time filter object. Object should be either " + "TimeInstantRangeFilter or LocalTimeRangeFilter."); } } private long getPeriodDeltaInDays(Period period) { return period.getDays(); } private Instant getDurationInstant(long duration) { return Instant.ofEpochMilli(duration); } private long getDurationDelta(Duration duration) { return duration.toMillis(); } }