Skip to content

Commit

Permalink
feat: add monaco-kubernetes editor for IntelliSense
Browse files Browse the repository at this point in the history
Signed-off-by: Delnat Wito <wito.delnat@gmail.com>
  • Loading branch information
WitoDelnat committed Mar 15, 2023
1 parent dd546a6 commit abdda83
Show file tree
Hide file tree
Showing 6 changed files with 334 additions and 12 deletions.
1 change: 1 addition & 0 deletions ui/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
"minimatch": "^3.1.2",
"moment": "^2.29.4",
"monaco-editor": "^0.33.0",
"monaco-kubernetes": "0.1.1",
"path": "^0.12.7",
"prop-types": "^15.8.1",
"react": "^16.9.3",
Expand Down
10 changes: 7 additions & 3 deletions ui/src/app/shared/components/monaco-editor.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import * as React from 'react';

import * as monacoEditor from 'monaco-editor';
import {configure, LanguageSettings} from 'monaco-kubernetes';

export interface EditorInput {
text: string;
Expand All @@ -11,7 +12,7 @@ export interface MonacoProps {
minHeight?: number;
vScrollBar: boolean;
editor?: {
options?: monacoEditor.editor.IEditorOptions;
options?: monacoEditor.editor.IEditorOptions & {settings?: LanguageSettings};
input: EditorInput;
getApi?: (api: monacoEditor.editor.IEditor) => any;
};
Expand All @@ -25,11 +26,13 @@ const DEFAULT_LINE_HEIGHT = 18;

const MonacoEditorLazy = React.lazy(() =>
import('monaco-editor').then(monaco => {
require('monaco-editor/esm/vs/basic-languages/yaml/yaml.contribution.js');

const component = (props: MonacoProps) => {
const [height, setHeight] = React.useState(0);

React.useEffect(() => {
configure(props.editor.options.settings);
}, [props.editor.options.settings]);

return (
<div
style={{
Expand All @@ -47,6 +50,7 @@ const MonacoEditorLazy = React.lazy(() =>
const editor = monaco.editor.create(el, {
...props.editor.options,
scrollBeyondLastLine: props.vScrollBar,
renderValidationDecorations: 'on',
scrollbar: {
handleMouseWheel: false,
vertical: props.vScrollBar ? 'visible' : 'hidden'
Expand Down
54 changes: 54 additions & 0 deletions ui/src/app/shared/components/yaml-editor/yaml-config.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
import {LanguageSettings} from 'monaco-kubernetes';

/**
* The configuration of the editor.
*/
export type EditorConfig = {
/**
* Whether to use monaco-kubernetes in the editor.
*
* Enhancements
*
* - Schema validation for both well-known resources
* (e.g. Deployment, Ingress, etc) and Argo CRDs. This
* includes autocomplete and other common IntelliSense.
* - Adds semantical validation which includes the most
* prevalent Kubernetes hardening guidelines and additional
* rules to prevent misconfigurations in Argo CD CRDs.
*
* @remark All heavy lifting is done in a web worker which
* leaves the main thread free for rendering the UI.
*/
useKubernetesEditor: boolean;

/**
* Configuration of monaco-kubernetes.
*/

settings?: LanguageSettings;
};

export const DEFAULT_EDITOR_CONFIG: EditorConfig = {
useKubernetesEditor: true,
settings: {
// @see https://github.com/kubeshop/monokle-core/blob/main/packages/validation/docs/configuration.md
validation: {
plugins: {
'yaml-syntax': true,
'kubernetes-schema': true,
'open-policy-agent': true,
'argo': true
},
rules: {
// @see https://github.com/kubeshop/monokle-core/blob/main/packages/validation/docs/core-plugins.md#open-policy-agent
'open-policy-agent/no-latest-image': 'warn',
'open-policy-agent/no-low-user-id': 'warn',
'open-policy-agent/no-low-group-id': 'warn',
'open-policy-agent/no-elevated-process': 'err',
'open-policy-agent/no-sys-admin': 'err',
'open-policy-agent/no-host-mounted-path': 'err',
'open-policy-agent/no-host-port-access': 'err'
}
}
}
};
7 changes: 5 additions & 2 deletions ui/src/app/shared/components/yaml-editor/yaml-editor.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import * as React from 'react';

import {Consumer} from '../../context';
import {MonacoEditor} from '../monaco-editor';
import {DEFAULT_EDITOR_CONFIG, EditorConfig} from './yaml-config';

const jsonMergePatch = require('json-merge-patch');
require('./yaml-editor.scss');
Expand All @@ -18,6 +19,7 @@ export class YamlEditor<T> extends React.Component<
onSave?: (patch: string, patchType: string) => Promise<any>;
onCancel?: () => any;
minHeight?: number;
config?: EditorConfig;
},
{
editing: boolean;
Expand All @@ -33,6 +35,7 @@ export class YamlEditor<T> extends React.Component<
public render() {
const props = this.props;
const yaml = props.input ? jsYaml.safeDump(props.input) : '';
const config = props.config ?? DEFAULT_EDITOR_CONFIG;

return (
<div className='yaml-editor'>
Expand Down Expand Up @@ -97,8 +100,8 @@ export class YamlEditor<T> extends React.Component<
minHeight={props.minHeight}
vScrollBar={props.vScrollbar}
editor={{
input: {text: yaml, language: 'yaml'},
options: {readOnly: !this.state.editing, minimap: {enabled: false}},
input: {text: yaml, language: config.useKubernetesEditor ? 'kubernetes' : 'yaml'},
options: {readOnly: !this.state.editing, minimap: {enabled: false}, settings: config.settings},
getApi: api => {
this.model = api.getModel() as monacoEditor.editor.ITextModel;
}
Expand Down
12 changes: 11 additions & 1 deletion ui/src/app/webpack.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,17 @@ const config = {
}),
new MonacoWebpackPlugin({
// https://github.com/microsoft/monaco-editor-webpack-plugin#options
languages: ['yaml']
languages: ['yaml'],
customLanguages: [
{
label: 'kubernetes',
entry: 'monaco-kubernetes',
worker: {
id: 'monaco-kubernetes/kubernetesWorker',
entry: 'monaco-kubernetes/worker'
}
}
]
})
],
devServer: {
Expand Down
Loading

0 comments on commit abdda83

Please sign in to comment.