mirror of
https://github.com/kemko/nomad.git
synced 2026-01-03 17:05:43 +03:00
* Hook and latch on the initial index * Serialization and restart of controller and table * de-log * allocBlocks reimplemented at job model level * totalAllocs doesnt mean on jobmodel what it did in steady.js * Hamburgers to sausages * Hacky way to bring new jobs back around and parent job handling in list view * Getting closer to hook/latch * Latch from update on hook from initialize, but fickle * Note on multiple-watch problem * Sensible monday morning comment removal * use of abortController to handle transition and reset events * Next token will now update when there's an on-page shift * Very rough anti-jostle technique * Demoable, now to move things out of route and into controller * Into the controller, generally * Smarter cancellations * Reset abortController on index models run, and system/sysbatch jobs now have an improved groupCountSum computed property * Prev Page reverse querying * n+1th jobs existing will trigger nextToken/pagination display * Start of a GET/POST statuses return * Namespace fix * Unblock tests * Realizing to my small horror that this skipURLModification flag may be too heavy handed * Lintfix * Default liveupdates localStorage setting to true * Pagination and index rethink * Big uncoupling of watchable and url-append stuff * Testfixes for region, search, and keyboard * Job row class for test purposes * Allocations in test now contain events * Starting on the jobs list tests in earnest * Forbidden state de-bubbling cleanup * Job list page size fixes * Facet/Search/Filter jobs list tests skipped * Maybe it's the automatic mirage logging * Unbreak task unit test * Pre-sort sort * styling for jobs list pagination and general PR cleanup * moving from Job.ActiveDeploymentID to Job.LatestDeployment.ID * modifyIndex-based pagination (#20350) * modifyIndex-based pagination * modifyIndex gets its own column and pagination compacted with icons * A generic withPagination handler for mirage * Some live-PR changes * Pagination and button disabled tests * Job update handling tests for jobs index * assertion timeout in case of long setTimeouts * assert.timeouts down to 500ms * de-to-do * Clarifying comment and test descriptions * Bugfix: resizing your browser on the new jobs index page would make the viz grow forever (#20458) * [ui] Searching and filtering options (#20459) * Beginnings of a search box for filter expressions * jobSearchBox integration test * jobs list updateFilter initial test * Basic jobs list filtering tests * First attempt at side-by-side facets and search with a computed filter * Weirdly close to an iterative approach but checked isnt tracked properly * Big rework to make filter composition and decomposition work nicely with the url * Namespace facet dropdown added * NodePool facet dropdown added * hdsFacet for future testing and basic namespace filtering test * Namespace filter existence test * Status filtering * Node pool/dynamic facet test * Test patchups * Attempt at optimize test fix * Allocation re-load on optimize page explainer * The Big Un-Skip * Post-PR-review cleanup * todo-squashing * [ui] Handle parent/child jobs with the paginated Jobs Index route (#20493) * First pass at a non-watchQuery version * Parameterized jobs get child fetching and jobs index status style for parent jobs * Completed allocs vs Running allocs in a child-job context, and fix an issue where moving from parent to parent would not reset index * Testfix and better handling empty-child-statuses-list * Parent/child test case * Dont show empty allocation-status bars for parent jobs with no children * Splits Settings into 2 sections, sign-in/profile and user settings (#20535) * Changelog
162 lines
4.4 KiB
JavaScript
162 lines
4.4 KiB
JavaScript
/**
|
|
* Copyright (c) HashiCorp, Inc.
|
|
* SPDX-License-Identifier: BUSL-1.1
|
|
*/
|
|
|
|
// @ts-check
|
|
|
|
import Ember from 'ember';
|
|
import { get } from '@ember/object';
|
|
import { assert } from '@ember/debug';
|
|
import RSVP from 'rsvp';
|
|
import { task } from 'ember-concurrency';
|
|
import { AbortController } from 'fetch';
|
|
import wait from 'nomad-ui/utils/wait';
|
|
import Watchable from 'nomad-ui/adapters/watchable';
|
|
import config from 'nomad-ui/config/environment';
|
|
|
|
const isEnabled = config.APP.blockingQueries !== false;
|
|
|
|
/**
|
|
* @typedef watchRecordOptions
|
|
* @property {boolean} [shouldSurfaceErrors=false] - If true, the task will throw errors instead of yielding them.
|
|
*/
|
|
|
|
/**
|
|
* @param {string} modelName - The name of the model to watch.
|
|
* @param {watchRecordOptions} [options]
|
|
*/
|
|
export function watchRecord(modelName, { shouldSurfaceErrors = false } = {}) {
|
|
return task(function* (id, throttle = 2000) {
|
|
assert(
|
|
'To watch a record, the record adapter MUST extend Watchable',
|
|
this.store.adapterFor(modelName) instanceof Watchable
|
|
);
|
|
if (typeof id === 'object') {
|
|
id = get(id, 'id');
|
|
}
|
|
while (isEnabled && !Ember.testing) {
|
|
const controller = new AbortController();
|
|
try {
|
|
yield RSVP.all([
|
|
this.store.findRecord(modelName, id, {
|
|
reload: true,
|
|
adapterOptions: { watch: true, abortController: controller },
|
|
}),
|
|
wait(throttle),
|
|
]);
|
|
} catch (e) {
|
|
if (shouldSurfaceErrors) {
|
|
throw e;
|
|
}
|
|
yield e;
|
|
break;
|
|
} finally {
|
|
controller.abort();
|
|
}
|
|
}
|
|
}).drop();
|
|
}
|
|
|
|
export function watchRelationship(relationshipName, replace = false) {
|
|
return task(function* (model, throttle = 2000) {
|
|
assert(
|
|
'To watch a relationship, the adapter of the model provided to the watchRelationship task MUST extend Watchable',
|
|
this.store.adapterFor(model.constructor.modelName) instanceof Watchable
|
|
);
|
|
while (isEnabled && !Ember.testing) {
|
|
const controller = new AbortController();
|
|
try {
|
|
yield RSVP.all([
|
|
this.store
|
|
.adapterFor(model.constructor.modelName)
|
|
.reloadRelationship(model, relationshipName, {
|
|
watch: true,
|
|
abortController: controller,
|
|
replace,
|
|
}),
|
|
wait(throttle),
|
|
]);
|
|
} catch (e) {
|
|
yield e;
|
|
break;
|
|
} finally {
|
|
controller.abort();
|
|
}
|
|
}
|
|
}).drop();
|
|
}
|
|
|
|
export function watchNonStoreRecords(modelName) {
|
|
return task(function* (model, asyncCallbackName, throttle = 5000) {
|
|
assert(
|
|
'To watch a non-store records, the adapter of the model provided to the watchNonStoreRecords task MUST extend Watchable',
|
|
this.store.adapterFor(modelName) instanceof Watchable
|
|
);
|
|
while (isEnabled && !Ember.testing) {
|
|
const controller = new AbortController();
|
|
try {
|
|
yield model[asyncCallbackName]();
|
|
yield wait(throttle);
|
|
} catch (e) {
|
|
yield e;
|
|
break;
|
|
} finally {
|
|
controller.abort();
|
|
}
|
|
}
|
|
}).drop();
|
|
}
|
|
|
|
export function watchAll(modelName) {
|
|
return task(function* (throttle = 2000) {
|
|
assert(
|
|
'To watch all, the respective adapter MUST extend Watchable',
|
|
this.store.adapterFor(modelName) instanceof Watchable
|
|
);
|
|
while (isEnabled && !Ember.testing) {
|
|
const controller = new AbortController();
|
|
try {
|
|
yield RSVP.all([
|
|
this.store.findAll(modelName, {
|
|
reload: true,
|
|
adapterOptions: { watch: true, abortController: controller },
|
|
}),
|
|
wait(throttle),
|
|
]);
|
|
} catch (e) {
|
|
yield e;
|
|
break;
|
|
} finally {
|
|
controller.abort();
|
|
}
|
|
}
|
|
}).drop();
|
|
}
|
|
|
|
export function watchQuery(modelName) {
|
|
return task(function* (params, throttle = 2000 /*options = {}*/) {
|
|
assert(
|
|
'To watch a query, the adapter for the type being queried MUST extend Watchable',
|
|
this.store.adapterFor(modelName) instanceof Watchable
|
|
);
|
|
while (isEnabled && !Ember.testing) {
|
|
const controller = new AbortController();
|
|
try {
|
|
yield RSVP.all([
|
|
this.store.query(modelName, params, {
|
|
reload: true,
|
|
adapterOptions: { watch: true, abortController: controller },
|
|
}),
|
|
wait(throttle),
|
|
]);
|
|
} catch (e) {
|
|
yield e;
|
|
break;
|
|
} finally {
|
|
controller.abort();
|
|
}
|
|
}
|
|
}).drop();
|
|
}
|