| /** |
| * @license |
| * Copyright (C) 2020 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. |
| */ |
| import {routerChangeNum$} from '../router/router-model'; |
| import {change$, updateState} from './change-model'; |
| import {ParsedChangeInfo} from '../../types/types'; |
| import {appContext} from '../app-context'; |
| import {ChangeInfo} from '../../types/common'; |
| import { |
| computeAllPatchSets, |
| computeLatestPatchNum, |
| } from '../../utils/patch-set-util'; |
| |
| export class ChangeService { |
| private change?: ParsedChangeInfo; |
| |
| private readonly restApiService = appContext.restApiService; |
| |
| constructor() { |
| // TODO: In the future we will want to make restApiService.getChangeDetail() |
| // calls from a switchMap() here. For now just make sure to invalidate the |
| // change when no changeNum is set. |
| routerChangeNum$.subscribe(changeNum => { |
| if (!changeNum) updateState(undefined); |
| }); |
| change$.subscribe(change => { |
| this.change = change; |
| }); |
| } |
| |
| /** |
| * This is a temporary indirection between change-view, which currently |
| * manages what the current change is, and the change-model, which will |
| * become the source of truth in the future. We will extract a substantial |
| * amount of code from change-view and move it into this change-service. This |
| * will take some time ... |
| */ |
| updateChange(change: ParsedChangeInfo) { |
| updateState(change); |
| } |
| |
| /** |
| * Typically you would just subscribe to change$ yourself to get updates. But |
| * sometimes it is nice to also be able to get the current ChangeInfo on |
| * demand. So here it is for your convenience. |
| */ |
| getChange() { |
| return this.change; |
| } |
| |
| /** |
| * Check whether there is no newer patch than the latest patch that was |
| * available when this change was loaded. |
| * |
| * @return A promise that yields true if the latest patch |
| * has been loaded, and false if a newer patch has been uploaded in the |
| * meantime. The promise is rejected on network error. |
| */ |
| fetchChangeUpdates(change: ChangeInfo | ParsedChangeInfo) { |
| const knownLatest = computeLatestPatchNum(computeAllPatchSets(change)); |
| return this.restApiService.getChangeDetail(change._number).then(detail => { |
| if (!detail) { |
| const error = new Error('Change detail not found.'); |
| return Promise.reject(error); |
| } |
| const actualLatest = computeLatestPatchNum(computeAllPatchSets(detail)); |
| if (!actualLatest || !knownLatest) { |
| const error = new Error('Unable to check for latest patchset.'); |
| return Promise.reject(error); |
| } |
| return { |
| isLatest: actualLatest <= knownLatest, |
| newStatus: change.status !== detail.status ? detail.status : null, |
| newMessages: |
| (change.messages || []).length < (detail.messages || []).length |
| ? detail.messages![detail.messages!.length - 1] |
| : undefined, |
| }; |
| }); |
| } |
| } |