Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[ILM] Policy form should not throw away data #83077

Merged
merged 12 commits into from
Nov 20, 2020
Merged
Original file line number Diff line number Diff line change
Expand Up @@ -195,3 +195,41 @@ export const POLICY_WITH_NODE_ROLE_ALLOCATION: PolicyFromES = {
},
name: POLICY_NAME,
};

export const POLICY_WITH_KNOWN_AND_UNKNOWN_FIELDS = ({
version: 1,
modified_date: Date.now().toString(),
policy: {
foo: 'bar',
phases: {
hot: {
min_age: '0ms',
actions: {
rollover: {
unknown_setting: 123,
max_size: '50gb',
},
},
},
warm: {
actions: {
my_unfollow_action: {},
set_priority: {
priority: 22,
unknown_setting: true,
},
},
},
delete: {
wait_for_snapshot: {
policy: SNAPSHOT_POLICY_NAME,
},
delete: {
delete_searchable_snapshot: true,
},
},
},
name: POLICY_NAME,
},
name: POLICY_NAME,
} as any) as PolicyFromES;
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import {
POLICY_WITH_INCLUDE_EXCLUDE,
POLICY_WITH_NODE_ATTR_AND_OFF_ALLOCATION,
POLICY_WITH_NODE_ROLE_ALLOCATION,
POLICY_WITH_KNOWN_AND_UNKNOWN_FIELDS,
getDefaultHotPhasePolicy,
} from './constants';

Expand All @@ -31,6 +32,70 @@ describe('<EditPolicy />', () => {
server.restore();
});

describe('serialization', () => {
/**
* We assume that policies that populate this form are loaded directly from ES and so
* are valid according to ES. There may be settings in the policy created through the ILM
* API that the UI does not cater for, like the unfollow action. We do not want to overwrite
* the configuration for these actions in the UI.
*/
it('preserves policy settings it did not configure', async () => {
httpRequestsMockHelpers.setLoadPolicies([POLICY_WITH_KNOWN_AND_UNKNOWN_FIELDS]);
httpRequestsMockHelpers.setLoadSnapshotPolicies([]);
httpRequestsMockHelpers.setListNodes({
nodesByRoles: {},
nodesByAttributes: { test: ['123'] },
isUsingDeprecatedDataRoleConfig: false,
});

await act(async () => {
testBed = await setup();
});

const { component, actions } = testBed;
component.update();

// Set max docs to test whether we keep the unknown fields in that object after serializing
await actions.hot.setMaxDocs('1000');
// Remove the delete phase to ensure that we also correctly remove data
await actions.delete.enable(false);
await actions.savePolicy();

const latestRequest = server.requests[server.requests.length - 1];
const entirePolicy = JSON.parse(JSON.parse(latestRequest.requestBody).body);

expect(entirePolicy).toEqual({
foo: 'bar', // Made up value
name: 'my_policy',
phases: {
hot: {
actions: {
rollover: {
max_docs: 1000,
max_size: '50gb',
unknown_setting: 123, // Made up setting that should stay preserved
},
set_priority: {
priority: 100,
},
},
min_age: '0ms',
},
warm: {
actions: {
my_unfollow_action: {}, // Made up action
set_priority: {
priority: 22,
unknown_setting: true,
},
},
min_age: '0ms',
},
},
});
});
});

describe('hot phase', () => {
describe('serialization', () => {
beforeEach(async () => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -298,12 +298,12 @@ describe('edit policy', () => {
phases: {
hot: {
actions: {
set_priority: {
priority: 100,
},
rollover: {
max_size: '50gb',
max_age: '30d',
max_size: '50gb',
},
set_priority: {
priority: 100,
},
},
min_age: '0ms',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,13 +22,11 @@ export const deserializer = (policy: SerializedPolicy): FormInternal => {
const _meta: FormInternal['_meta'] = {
hot: {
useRollover: Boolean(hot?.actions?.rollover),
forceMergeEnabled: Boolean(hot?.actions?.forcemerge),
bestCompression: hot?.actions?.forcemerge?.index_codec === 'best_compression',
},
warm: {
enabled: Boolean(warm),
warmPhaseOnRollover: Boolean(warm?.min_age === '0ms'),
forceMergeEnabled: Boolean(warm?.actions?.forcemerge),
bestCompression: warm?.actions?.forcemerge?.index_codec === 'best_compression',
dataTierAllocationType: determineDataTierAllocationType(warm?.actions),
},
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,183 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License;
* you may not use this file except in compliance with the Elastic License.
*/

import { setAutoFreeze } from 'immer';
import { cloneDeep } from 'lodash';
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shouldn't we use the lodash.clonedeep package?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think not after #78156 was merged. We should be importing the entire lodash library now.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could. Thanks, good to know 👍

import { SerializedPolicy } from '../../../../../common/types';
import { deserializer } from './deserializer';
import { createSerializer } from './serializer';
import { FormInternal } from '../types';

const isObject = (v: unknown): v is { [key: string]: any } =>
Object.prototype.toString.call(v) === '[object Object]';

const unknownValue = { some: 'value' };

const populateWithUnknownEntries = (v: unknown) => {
if (isObject(v)) {
for (const key of Object.keys(v)) {
if (key === 'require' || key === 'include' || key === 'exclude') continue; // this will generate an invalid policy
Copy link
Contributor

@cjcenizal cjcenizal Nov 19, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for adding this helpful comment! This might be a bit easier to read (not a blocker though):

if (['require', 'include', 'exclude'].includes(key)) continue;

populateWithUnknownEntries(v[key]);
}
v.unknown = unknownValue;
return;
}
if (Array.isArray(v)) {
v.forEach(populateWithUnknownEntries);
}
};

const originalPolicy: SerializedPolicy = {
name: 'test',
phases: {
hot: {
actions: {
rollover: {
max_age: '1d',
max_size: '10gb',
max_docs: 1000,
},
forcemerge: {
index_codec: 'best_compression',
max_num_segments: 22,
},
set_priority: {
priority: 1,
},
},
min_age: '12ms',
},
warm: {
actions: {
shrink: { number_of_shards: 12 },
allocate: {
number_of_replicas: 3,
},
set_priority: {
priority: 10,
},
migrate: { enabled: false },
},
},
cold: {
actions: {
allocate: {
number_of_replicas: 12,
require: { test: 'my_value' },
include: { test: 'my_value' },
exclude: { test: 'my_value' },
},
freeze: {},
set_priority: {
priority: 12,
},
},
},
delete: {
actions: {
delete: {
delete_searchable_snapshot: true,
},
wait_for_snapshot: {
policy: 'test',
},
},
},
},
};

describe('deserializer and serializer', () => {
let policy: SerializedPolicy;
let serializer: ReturnType<typeof createSerializer>;
let formInternal: FormInternal;

// So that we can modify produced form objects
beforeAll(() => setAutoFreeze(false));
// This is the default in dev, so change back to true (https://github.com/immerjs/immer/blob/master/docs/freezing.md)
afterAll(() => setAutoFreeze(true));

beforeEach(() => {
policy = cloneDeep(originalPolicy);
serializer = createSerializer(policy);
formInternal = deserializer(policy);
});

it('preserves any unknown policy settings', () => {
// We populate all levels of the policy with entries our UI does not know about
populateWithUnknownEntries(policy);

const copyOfPolicy = cloneDeep(policy);

expect(serializer(formInternal)).toEqual(policy);

// Assert that the policy we passed in is unaltered after deserialization and serialization
expect(policy).toEqual(copyOfPolicy);
});

it('removes all phases if they were disabled in the form', () => {
formInternal._meta.warm.enabled = false;
formInternal._meta.cold.enabled = false;
formInternal._meta.delete.enabled = false;

expect(serializer(formInternal)).toEqual({
...policy,
Copy link
Contributor

@cjcenizal cjcenizal Nov 19, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This line makes me read the definition of policy to understand what we're spreading and what will be retained. But it looks like the only thing we're retaining is the name property. I would find this easier to read if we didn't spread, and instead just hardcoded the name. WDYT? Just a suggestion, not a blocker.

    expect(serializer(formInternal)).toEqual({
      name: 'test',
      phases: {
        hot: policy.phases.hot, // We expect to see only the hot phase
      },
    });

phases: {
hot: policy.phases.hot, // We expect to see only the hot phase
},
});
});

it('removes the forcemerge action if it is disabled in the form', () => {
delete formInternal.phases.hot!.actions.forcemerge;
delete formInternal.phases.warm!.actions.forcemerge;

const result = serializer(formInternal);

expect(result.phases.hot!.actions.forcemerge).toBeUndefined();
expect(result.phases.warm!.actions.forcemerge).toBeUndefined();
});

it('removes set priority if it is disabled in the form', () => {
delete formInternal.phases.hot!.actions.set_priority;
delete formInternal.phases.warm!.actions.set_priority;
delete formInternal.phases.cold!.actions.set_priority;

const result = serializer(formInternal);

expect(result.phases.hot!.actions.set_priority).toBeUndefined();
expect(result.phases.warm!.actions.set_priority).toBeUndefined();
expect(result.phases.cold!.actions.set_priority).toBeUndefined();
});

it('removes freeze setting in the cold phase if it is disabled in the form', () => {
formInternal._meta.cold.freezeEnabled = false;

const result = serializer(formInternal);

expect(result.phases.cold!.actions.freeze).toBeUndefined();
});

it('removes node attribute allocation when it is not selected in the form', () => {
// Change from 'node_attrs' to 'node_roles'
formInternal._meta.cold.dataTierAllocationType = 'node_roles';

const result = serializer(formInternal);

expect(result.phases.cold!.actions.allocate!.number_of_replicas).toBe(12);
expect(result.phases.cold!.actions.allocate!.require).toBeUndefined();
expect(result.phases.cold!.actions.allocate!.include).toBeUndefined();
expect(result.phases.cold!.actions.allocate!.exclude).toBeUndefined();
});

it('removes forcemerge and rollover config when rollover is disabled in hot phase', () => {
formInternal._meta.hot.useRollover = false;

const result = serializer(formInternal);

expect(result.phases.hot!.actions.rollover).toBeUndefined();
expect(result.phases.hot!.actions.forcemerge).toBeUndefined();
});
});
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ import { i18nTexts } from '../i18n_texts';
const { emptyField, numberGreaterThanField } = fieldValidators;

const serializers = {
stringToNumber: (v: string): any => (v ? parseInt(v, 10) : undefined),
stringToNumber: (v: string): any => (v != null ? parseInt(v, 10) : undefined),
};

export const schema: FormSchema<FormInternal> = {
Expand Down
Loading