Skip to content

Commit

Permalink
more
Browse files Browse the repository at this point in the history
  • Loading branch information
oatkiller committed Apr 17, 2020
1 parent 2e606cb commit 85722d8
Show file tree
Hide file tree
Showing 7 changed files with 73 additions and 44 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,15 @@

import { IIndexPattern } from 'src/plugins/data/public';
import { AlertResultList, AlertDetails } from '../../../../../common/types';
import { MiddlewareFactory, AlertListState } from '../../types';
import { ImmutableMiddlewareFactory, AlertListState } from '../../types';
import { isOnAlertPage, apiQueryParams, hasSelectedAlert, uiQueryParams } from './selectors';
import { cloneHttpFetchQuery } from '../../../../common/clone_http_fetch_query';
import { EndpointAppConstants } from '../../../../../common/types';

export const alertMiddlewareFactory: MiddlewareFactory<AlertListState> = (coreStart, depsStart) => {
export const alertMiddlewareFactory: ImmutableMiddlewareFactory<AlertListState> = (
coreStart,
depsStart
) => {
async function fetchIndexPatterns(): Promise<IIndexPattern[]> {
const { indexPatterns } = depsStart.data;
const eventsPattern: { indexPattern: string } = await coreStart.http.get(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,11 @@
* you may not use this file except in compliance with the Elastic License.
*/

import { MiddlewareFactory } from '../../types';
import { ImmutableMiddlewareFactory } from '../../types';
import { pageIndex, pageSize, isOnHostPage, hasSelectedHost, uiQueryParams } from './selectors';
import { HostListState } from '../../types';

export const hostMiddlewareFactory: MiddlewareFactory<HostListState> = coreStart => {
export const hostMiddlewareFactory: ImmutableMiddlewareFactory<HostListState> = coreStart => {
return ({ getState, dispatch }) => next => async action => {
next(action);
const state = getState();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,45 +4,25 @@
* you may not use this file except in compliance with the Elastic License.
*/

import {
createStore,
compose,
applyMiddleware,
Store,
MiddlewareAPI,
Dispatch,
Middleware,
} from 'redux';
import { createStore, compose, applyMiddleware, Store } from 'redux';
import { CoreStart } from 'kibana/public';
import { appReducer } from './reducer';
import { alertMiddlewareFactory } from './alerts/middleware';
import { hostMiddlewareFactory } from './hosts';
import { policyListMiddlewareFactory } from './policy_list';
import { policyDetailsMiddlewareFactory } from './policy_details';
import { GlobalState, MiddlewareFactory } from '../types';
import { AppAction } from './action';
import { ImmutableMiddlewareFactory, SubstateMiddlewareFactory } from '../types';
import { EndpointPluginStartDependencies } from '../../../plugin';
import { Immutable } from '../../../../common/types';

const composeWithReduxDevTools = (window as any).__REDUX_DEVTOOLS_EXTENSION_COMPOSE__
? (window as any).__REDUX_DEVTOOLS_EXTENSION_COMPOSE__({ name: 'EndpointApp' })
: compose;

export type Selector<S, R> = (state: S) => R;

/**
* Wrap Redux Middleware and adjust 'getState()' to return the namespace from 'GlobalState that applies to the given Middleware concern.
*
* @param selector
* @param middleware
*/
export const substateMiddlewareFactory = <Substate>(
selector: Selector<GlobalState, Substate>,
middleware: Middleware<{}, Substate, Dispatch<Immutable<AppAction> | AppAction>>
): Middleware<{}, GlobalState, Dispatch<AppAction | Immutable<AppAction>>> => {
export const substateMiddlewareFactory: SubstateMiddlewareFactory = (selector, middleware) => {
return api => {
const substateAPI: MiddlewareAPI<Dispatch<AppAction | Immutable<AppAction>>, Substate> = {
const substateAPI = {
...api,
// Return just the substate instead of global state.
getState() {
return selector(api.getState());
},
Expand All @@ -67,7 +47,7 @@ export const appStoreFactory: (middlewareDeps?: {
* Any additional Redux Middlewares
* (should only be used for testing - example: to inject the action spy middleware)
*/
additionalMiddleware?: Array<ReturnType<MiddlewareFactory>>;
additionalMiddleware?: Array<ReturnType<ImmutableMiddlewareFactory>>;
}) => Store = middlewareDeps => {
let middleware;
if (middlewareDeps) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
* you may not use this file except in compliance with the Elastic License.
*/

import { MiddlewareFactory, PolicyDetailsState, UpdatePolicyResponse } from '../../types';
import { ImmutableMiddlewareFactory, PolicyDetailsState, UpdatePolicyResponse } from '../../types';
import { policyIdFromParams, isOnPolicyDetailsPage, policyDetails } from './selectors';
import {
sendGetDatasource,
Expand All @@ -14,7 +14,7 @@ import {
import { PolicyData } from '../../../../../common/types';
import { factory as policyConfigFactory } from '../../../../../common/models/policy_config';

export const policyDetailsMiddlewareFactory: MiddlewareFactory<PolicyDetailsState> = coreStart => {
export const policyDetailsMiddlewareFactory: ImmutableMiddlewareFactory<PolicyDetailsState> = coreStart => {
const http = coreStart.http;

return ({ getState, dispatch }) => next => async action => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,11 @@
* you may not use this file except in compliance with the Elastic License.
*/

import { GetPolicyListResponse, MiddlewareFactory, PolicyListState } from '../../types';
import { GetPolicyListResponse, ImmutableMiddlewareFactory, PolicyListState } from '../../types';
import { sendGetEndpointSpecificDatasources } from './services/ingest';
import { isOnPolicyListPage, urlSearchParams } from './selectors';

export const policyListMiddlewareFactory: MiddlewareFactory<PolicyListState> = coreStart => {
export const policyListMiddlewareFactory: ImmutableMiddlewareFactory<PolicyListState> = coreStart => {
const http = coreStart.http;

return ({ getState, dispatch }) => next => async action => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
*/

import { Dispatch } from 'redux';
import { AppAction, GlobalState, MiddlewareFactory } from '../types';
import { AppAction, GlobalState, ImmutableMiddlewareFactory } from '../types';

/**
* Utilities for testing Redux middleware
Expand Down Expand Up @@ -35,7 +35,7 @@ export interface MiddlewareActionSpyHelper<S = GlobalState, A extends AppAction
/**
* Redux middleware that enables spying on the action that are dispatched through the store
*/
actionSpyMiddleware: ReturnType<MiddlewareFactory<S>>;
actionSpyMiddleware: ReturnType<ImmutableMiddlewareFactory<S>>;
}

/**
Expand Down Expand Up @@ -109,7 +109,7 @@ export const createSpyMiddleware = <
return spyDispatch.mock;
},

actionSpyMiddleware: api => {
actionSpyMiddleware: () => {
return next => {
spyDispatch = jest.fn(action => {
next(action);
Expand Down
60 changes: 53 additions & 7 deletions x-pack/plugins/endpoint/public/applications/endpoint/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,13 @@
* you may not use this file except in compliance with the Elastic License.
*/

import { Dispatch, MiddlewareAPI, Action as ReduxAction, AnyAction as ReduxAnyAction } from 'redux';
import {
Dispatch,
Action as ReduxAction,
AnyAction as ReduxAnyAction,
Action,
Middleware,
} from 'redux';
import { IIndexPattern } from 'src/plugins/data/public';
import {
HostMetadata,
Expand All @@ -29,18 +35,58 @@ import {

export { AppAction };

/**
* like redux's `MiddlewareAPI` but `getState` returns an `Immutable` version of
* state and `dispatch` accepts `Immutable` versions of actions.
*/
export interface ImmutableMiddlewareAPI<S, A extends Action> {
dispatch: Dispatch<A | Immutable<A>>;
getState(): Immutable<S>;
}

/**
* Like redux's `Middleware` but without the ability to mutate actions or state.
* Differences:
* * `getState` returns an `Immutable` version of state
* * `dispatch` accepts `Immutable` versions of actions
* * `action`s received will be `Immutable`
*/
export type ImmutableMiddleware<S, A extends Action> = (
api: ImmutableMiddlewareAPI<S, A>
) => (next: Dispatch<A | Immutable<A>>) => (action: Immutable<A>) => unknown;

/**
* Takes application-standard middleware dependencies
* and returns a redux middleware.
* Middleware will be of the `ImmutableMiddleware` variety. Not able to directly
* change actions or state.
*/
export type MiddlewareFactory<S = GlobalState> = (
export type ImmutableMiddlewareFactory<S = GlobalState> = (
coreStart: CoreStart,
depsStart: EndpointPluginStartDependencies
) => (
api: MiddlewareAPI<Dispatch<Immutable<AppAction> | AppAction>, Immutable<S>>
) => (
next: Dispatch<Immutable<AppAction> | AppAction>
) => (action: Immutable<AppAction>) => unknown;
) => ImmutableMiddleware<S, AppAction>;

/**
* Simple type for a redux selector.
*/
type Selector<S, R> = (state: S) => R;

/**
* Takes a selector and an `ImmutableMiddleware`. The
* middleware's version of `getState` will receive
* the result of the selector instead of the global state.
*
* This allows middleware to have knowledge of only a subsection of state.
*
* `selector` returns an `Immutable` version of the substate.
* `middleware` must be an `ImmutableMiddleware`.
*
* Returns a regular middleware, meant to be used with `applyMiddleware`.
*/
export type SubstateMiddlewareFactory = <Substate>(
selector: Selector<GlobalState, Immutable<Substate>>,
middleware: ImmutableMiddleware<Substate, AppAction>
) => Middleware<{}, GlobalState, Dispatch<AppAction | Immutable<AppAction>>>;

export interface HostListState {
hosts: HostMetadata[];
Expand Down

0 comments on commit 85722d8

Please sign in to comment.