88 | {/*
*/}
89 | {['right'].map((anchor) => (
90 |
91 | Menu
92 |
97 | {list(anchor)}
98 |
99 |
100 | ))}
101 |
102 |
103 |
104 |
105 |
106 |
107 | );
108 | }
109 | export default withRouter(Navigation);
--------------------------------------------------------------------------------
/public/client/Components/LogRowComponent.jsx:
--------------------------------------------------------------------------------
1 | import React, {useEffect, useState} from 'react';
2 | import { connect } from 'react-redux';
3 | import { DataGrid } from '@mui/x-data-grid';
4 | import { makeStyles } from '@mui/styles';
5 | import { MailRounded } from '@mui/icons-material';
6 | import { minHeight } from '@mui/system';
7 |
8 | const mapStateToProps = (state) => {
9 | return {
10 | appLogs: state.logsReducer.appLogs,
11 | }
12 | }
13 | const columns = [
14 | {field: 'message', headerName: 'message',minWidth:150, headerAlign:'left', flex:1},
15 | {field: 'timestamp', headerName: 'timestamp', minWidth:100,headerAlign:'left', flex:1},
16 | {field: 'podName', headerName: 'pod name', minWidth:150,headerAlign:'left', flex:1},
17 | {field: 'host', headerName: 'host',headerAlign:'left',minWidth:150,flex:1},
18 | {field: 'level', headerName: 'level',headerAlign:'left',minWidth:50,flex:.5},
19 | {field: 'id', headerName: 'id', headerAlign:'left',minWidth:100,flex:1},
20 | ];
21 |
22 | const useStyles = makeStyles({
23 | root:{
24 | '& .MuiDataGrid-renderingZone': {
25 | maxHeight: 'none !important',
26 | maxWidth:'100% !important',
27 | margin: '0px !important',
28 | padding: '0px !important',
29 | },
30 | '& .MuiDataGrid-cell': {
31 | lineHeight: 'unset !important',
32 | overflowWrap: 'break-word !important',
33 | wordWrap: 'break-word !important',
34 | maxHeight: 'none !important',
35 | whiteSpace: 'normal',
36 | margin: '0px 0px 0px 0px!important',
37 | padding: '0px 8px 0px 8px !important'
38 | },
39 |
40 | '& .MuiDataGrid-columnsContainer': {
41 | backgroundColor: 'whitesmoke',
42 | borderRadius: 4,
43 | margin: '0px 0px 20px 0px !important',
44 | },
45 | '& .MuiDataGrid-columnHeader': {
46 |
47 | alignItems: 'flex-start !important',
48 | margin: '0px !important',
49 | padding: '0px !important',
50 | },
51 | '& .MuiDataGrid-columnHeaderTitleContainer': {
52 | alignItems: 'flex-start !important',
53 | justifyContent: 'flex-start !important',
54 | color: 'grey !important',
55 | margin: '0px !important',
56 | padding: '0px 0px 0px 10px !important',
57 | },
58 | '& .MuiDataGrid-columnHeaderDraggableContainer': {
59 | alignItems: 'flex-start !important',
60 | margin: '0px !important',
61 | padding: '0px !important',
62 | },
63 |
64 | '& .MuiDataGrid-row': {
65 | maxHeight: 'none !important',
66 | margin: '0px !important',
67 | padding: '0px !important',
68 | width:'100% !important',
69 |
70 | },
71 | '& .MuiDataGrid-window': {
72 | borderRadius: 4,
73 | margin: '1px 0px 0px 0px !important',
74 | overflowX: 'hidden !important',
75 | },
76 | '& .MuiDataGrid-viewport': {
77 | maxWidth: '100% !important',
78 | minWidth: '100% !important',
79 |
80 | },
81 | '& .MuiDataGrid-dataContainer': {
82 | margin: '0px !important',
83 | padding: '0px !important',
84 | },
85 | background: 'rgba(69,172,120,0.52)',
86 | border: 0,
87 | borderRadius: 4,
88 | //boxShadow: '0 3px 5px 2px rgba(255, 105, 135, .3)',
89 | color: 'white',
90 | height: '100%',
91 | width: '100%',
92 | padding: '0 30px',
93 | minHeight:480,
94 | overflowY:'auto',
95 | },
96 | footerContainer: {
97 | height: "1px !important",
98 | borderRadius: 4,
99 | background: 'rgba(69,172,120,0.52)',
100 | }
101 | })
102 |
103 |
104 | const LogRowComponent = (props) => {
105 | const classes = useStyles();
106 |
107 |
108 | return (
109 |
110 |
111 |
118 |
119 | );
120 | }
121 |
122 | export default connect(mapStateToProps, null)(LogRowComponent);
123 |
--------------------------------------------------------------------------------
/public/client/Containers/MetricsContainer.jsx:
--------------------------------------------------------------------------------
1 | import React,{useEffect} from 'react';
2 | import { connect } from 'react-redux';
3 | import * as actions from '../actions/metricsActionCreators.js';
4 | import NodeXContainer from './NodeXContainer.jsx';
5 | import NodeChartContainer from './NodeChartContainer.jsx';
6 | import PodChartContainer from './PodChartContainer.jsx';
7 | import PodsContainer from './PodsContainer.jsx';
8 | import MasterNodeContainer from './MasterNodeContainer.jsx';
9 | import { Container } from '@mui/material';
10 | import { makeStyles } from '@mui/styles';
11 | import Tabs from '@mui/material/Tabs';
12 | import Tab from '@mui/material/Tab';
13 | import Typography from '@mui/material/Typography';
14 | import Box from '@mui/material/Box';
15 |
16 | const mapStateToProps = state => {
17 | return {
18 | nodes: state.nodesReducer.nodes,
19 | }
20 | }
21 |
22 | const mapDispatchToProps = dispatch => {
23 | return {
24 | fetchNodeMetrics: () => dispatch(actions.fetchNodeMetrics()),
25 | }
26 | }
27 |
28 | const useStyles = makeStyles({
29 | root:{
30 | background: 'rgba(69,172,120,0.52)',
31 | border: 0,
32 | borderRadius: 4,
33 | boxShadow: '6px 2px 3px -1px rgba(0,0,0,0.75)',
34 | color: 'white',
35 | padding: '30px 30px 30px 30px',
36 | },
37 | })
38 |
39 | const tabStyles = makeStyles({
40 | flexContainer:{
41 | width:"100%",
42 | display:"flex",
43 | flexDirection:"row",
44 | justifyContent:'space-evenly'
45 | },
46 | scroller:{
47 | width:"100%",
48 | display:"flex",
49 | flexDirection:"row",
50 | justifyContent:'space-evenly'
51 | }
52 | })
53 |
54 | function TabPanel(props) {
55 | const { children, value, index, ...other } = props;
56 | return (
57 |
65 | {value === index && (
66 |
67 | {children}
68 |
69 | )}
70 |
71 | );
72 | }
73 |
74 | function addProps(index) {
75 | return {
76 | id: `cluster-tab-${index}`,
77 | 'aria-controls': `cluster-tabpanel-${index}`,
78 | };
79 | }
80 |
81 | function MetricsContainer(props) {
82 | const containerClasses = useStyles();
83 | const tabClasses = tabStyles();
84 | const [value, setValue] = React.useState(0);
85 |
86 | const handleChange = (event, newValue) => {
87 | setValue(newValue);
88 | };
89 |
90 | useEffect(() => {
91 | props.fetchNodeMetrics();
92 | }, []);
93 |
94 | const tabPanels = [];
95 | const tabs = [];
96 | let tabNum = 2;
97 |
98 | for (let node in props.nodes) {
99 | tabs.push(
);
100 | tabPanels.push(
101 |
105 |
106 |
107 | );
108 | tabNum += 1;
109 | }
110 |
111 | return (
112 |
113 |
119 |
120 |
121 |
122 |
123 | {tabs}
124 |
125 |
126 |
132 |
133 |
134 |
135 |
136 |
137 |
138 |
139 |
140 |
141 | {/* Tab Panel for each node */}
142 | {tabPanels}
143 |
144 |
145 |
146 | )
147 | }
148 |
149 | export default connect(mapStateToProps, mapDispatchToProps)(MetricsContainer);
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Kubric
2 |
3 | Kubric aims to provide a clean dashboard that displays important worker node and pod metrics. Kubric provides insight into the master node, the gatekeeper of communication to the cluster and responsible for orchestrating all the heavy lifting to corresponding worker nodes
4 |
5 | Additionally, Kubric persists logs and allows developers to query persisted logs even if a pod has been evicted and replaced. Developers need not worry about logs dying with pods or about log rotation policies because logs are persistently stored and queryable through Kubric
6 |
7 | * **Query Persistent Log Storage by Index Name, Field, Value**
8 |
9 |
10 |
11 |
12 |
13 | * **Toggle Visualization to View Relative Performance**
14 |
15 |
16 |
17 |
18 |
19 | * **Tab between Overview, Master and Worker Nodes Views**
20 |
21 |
22 |
23 |
24 |
25 | ## Set Up Prerequisites
26 |
27 | Here are links to the technology we used to implement our application. Be sure to have kubernetes installed before beginning setup
28 |
29 | * **Kubernetes**
30 | * To run commands against your cluster, make sure you have [Kubectl](https://kubernetes.io/docs/tasks/tools/) installed for your operating system
31 |
32 | * **Helm**
33 | * [Helm charts](https://helm.sh/docs/intro/install/) are a great resource to download interdependent YAML configuration files for complicated setup
34 |
35 | * **Fluentd**
36 | * [Fluentd](https://github.com/bitnami/charts/tree/master/bitnami/fluentd) is our log forwarding agent of choice
37 |
38 | * **Elasticsearch**
39 | * [Elasticsearch](https://github.com/elastic/helm-charts/tree/master/elasticsearch) is what we use for provisioning remote storage
40 |
41 | * **Prometheus**
42 | * [Prometheus](https://prometheus-operator.dev/docs/prologue/quick-start/) is the standard for metrics pipeline monitoring
43 |
44 | * **Linode LKE**
45 | * [Linode LKE](https://www.linode.com/) is the remote storage provider we chose, future support for GKE and EKS is in the works
46 |
47 | ## Set Up
48 |
49 | 1. Begin by cloning this repo
50 |
51 | 2. Create an account and provision three worker nodes with at least 8GB of RAM and 160GB of storage
52 |
53 | 3. Upon successful provisioning, download your Kubeconfig yaml file
54 |
55 | From the command line run:
56 | MacOS/Linux:
57 | ```
58 | export KUBECONFIG=/path/to/config.yaml
59 | ```
60 |
61 | Windows:
62 | ```
63 | create folder C:\Users\username\.kube
64 |
65 | rename cluster-config.yaml to config
66 |
67 | put config file in .kube folder
68 | ```
69 |
70 | You should now be connected to your remote cluster and able to run kubectl commands against it
71 |
72 | 4. To deploy our sample log generator app to the cluster:
73 |
74 | ```
75 | kubectl apply -f logGen-app/logGen-app-depl.yaml
76 | ```
77 |
78 | If you have not installed Helm do so now
79 |
80 | 5. Install Elasticsearch:
81 |
82 | ```
83 | helm repo add elastic https://Helm.elastic.co
84 | helm install elasticsearch elastic/elasticsearch -f values.yaml
85 | ```
86 |
87 | 6. Install Fluentd:
88 |
89 | ```
90 | helm repo add bitnami https://charts.bitnami.com/bitnami
91 | helm install fluentd bitnami/fluentd
92 | ```
93 |
94 | 7. Apply our log forwarding config file:
95 |
96 | ```
97 | kubectl apply -f fluent-update.yaml
98 | kubectl rollout restart daemonset/fluentd
99 | ```
100 |
101 | 8. Install Prometheus:
102 |
103 | To deploy prometheus to this cluster, follow the above link's quick start guide sections
104 |
105 | 8. To open ports for app access:
106 |
107 | ```
108 | kubectl --n monitoring port-forward svc/prometheus-k8s 9090
109 | kubectl port-forward service/elasticsearch-master 9200
110 | ```
111 |
112 |
113 |
114 | That's it!
115 | Make sure to npm install and for now npm run dev, navigate to localhost:8080 to log in and view cluster info.
116 |
117 | To add new applications to filter logs from, simply add another match statement to the fluentd config file and follow the syntax provided. Then kubectl rollout the update
118 |
119 |
120 | ## What's Next?
121 |
122 | **Kubric is still under the development**, further features and optimizations to be implemented! Stay tuned for more updates!
123 | Feel free to contact us for any comments, concerns, suggestions!
124 |
125 | ## Contributers
126 |
127 | * Laura Botel : [Github](https://github.com/laurabotel) | [LinkedIn](https://www.linkedin.com/in/laurabotel/)
128 | * Luke Cho : [Github](https://github.com/luke-h-cho) | [LinkedIn](https://www.linkedin.com/in/luke-h-cho/)
129 | * James Cross : [Github](https://github.com/James-P-Cross) | [LinkedIn](www.linkedin.com/in/james-p-cross1)
130 | * John Haberstroh : [Github](https://github.com/jlhline) | [LinkedIn](https://www.linkedin.com/in/john-haberstroh-9436ab117/)
131 |
132 |
133 |
--------------------------------------------------------------------------------
/public/client/scss/application.scss:
--------------------------------------------------------------------------------
1 | @import "_variables";
2 | @import url('https://fonts.googleapis.com/css2?family=Roboto&display=swap');
3 |
4 | html {
5 | font-family: 'Roboto', sans-serif;
6 | color: whitesmoke;
7 | // background-color: #fff;
8 | // background-image:
9 | // linear-gradient(90deg, transparent 79px, #abced4 79px, #abced4 81px, transparent 81px),
10 | // linear-gradient(#eee .1em, transparent .1em);
11 | // background-size: 100% 1.2em;
12 | }
13 |
14 | body {
15 | background: rgba(69, 172, 120, 0.52);
16 |
17 | }
18 |
19 | div {
20 |
21 | margin-top: .1em;
22 | margin-bottom: .1em;
23 | padding-top: .1em;
24 | padding-bottom: .1em;
25 | }
26 |
27 | input {
28 | margin: .25em;
29 | }
30 |
31 | button {
32 | margin: .25em;
33 | }
34 |
35 | #loginButtons {
36 | color: black
37 | }
38 | #backgroundContainer {
39 |
40 | }
41 |
42 | #loginPage {
43 | display:flex;
44 | flex-direction: column;
45 | }
46 |
47 | #loginPage h1 {
48 | color: black;
49 | }
50 |
51 | #user-pw {
52 | display: flex;
53 | flex-direction: column;
54 | }
55 |
56 | .MuiFormControl-root MuiTextField-root css-1u3bzj6-MuiFormControl-root-MuiTextField-root {
57 | align-content: center;
58 | align-items: center
59 | }
60 |
61 | #username {
62 | display: flex;
63 | align-items: center;
64 | align-content: center;
65 | }
66 |
67 | #login-container {
68 | // background-color: #F1FAEE;
69 | padding: .5em;
70 | }
71 | #metricsContainer {
72 | // background-color: #F1FAEE;
73 | padding: .5em;
74 | width: 85%;
75 | height: 100%;
76 | }
77 |
78 | #loginButtons {
79 | color: rgb(66, 65, 65);
80 | }
81 |
82 | #loginButtonIn {
83 | // background-color: rgb(180, 98, 114);
84 | border-style: solid;
85 | border-width: 2px;
86 | border-radius: 10px;
87 | border-color: rgb(241, 181, 226);
88 | font-size: 20px;
89 | }
90 |
91 | #loginButtonUp {
92 | border-style: solid;
93 | border-radius: 10px;
94 | border-width: 2px;
95 | border-color: rgb(181, 205, 241);
96 | font-size: 20px;
97 | }
98 |
99 | #appBarBox{
100 | // background-color: #F1FAEE;
101 | margin:auto;
102 | align-items:center;
103 | justify-content: center;
104 | padding: .5em;
105 | width: 85%;
106 | height: 100%;
107 | border-radius: 10,
108 | }
109 | #backgroundBox{
110 | // background-color: #F1FAEE;
111 | margin:auto;
112 | align-items:center;
113 | justify-content: center;
114 | padding: .5em;
115 | width: 85%;
116 | height: 100%;
117 | border-radius: 10;
118 | }
119 | #config-container {
120 | // background-color: #89B0AE;
121 | padding: .5em;
122 | }
123 |
124 |
125 | // #pods-container {
126 | // // background-color: #457B9D;
127 | // border-color: #457B9D;
128 | // background-color: #457B9D;
129 | // border-width: .25em;
130 | // border-style: solid;
131 | // padding: 5.5em 3.5em 3.5em;
132 | // display: flex;
133 | // flex-wrap: wrap;
134 | // // align-items: center;
135 | // justify-content: center;
136 | // align-items: center;
137 | // height: 300px;
138 | // width: 300px;
139 | // clip-path: polygon(50% 0%, 90% 20%, 100% 60%, 75% 100%, 25% 100%, 0% 60%, 10% 20%);
140 | // }
141 |
142 | // .pod-component {
143 | // background-color: rgb(49, 113, 49);
144 | // border-radius: 2.5%;
145 | // color: whitesmoke;
146 | // font-weight: bold;
147 | // text-align: center;
148 |
149 | // font-size: .75vh;
150 | // vertical-align: middle;
151 | // height: 5em;
152 | // width: 7.5em;
153 | // margin: .5em;
154 | // padding: .1em;
155 | // // clip-path: polygon(50% 0%, 90% 20%, 100% 60%, 75% 100%, 25% 100%, 0% 60%, 10% 20%);
156 | // }
157 |
158 | // #query-container {
159 | // // background-color: #A8DADC;
160 | // padding: .5em;
161 | // }
162 |
163 | #logContainer {
164 | // background-color: #A8DADC;
165 | padding: .5em;
166 | width: 85%;
167 | height: 100%;
168 |
169 | }
170 |
171 | #loginLogo h1 {
172 | text-align: center;
173 | }
174 |
175 | #loginPage #loginButtons {
176 | justify-content: center;
177 | }
178 |
179 | #loginPage #loginButtonIn {
180 | cursor: pointer;
181 | font-weight: 600;
182 | }
183 | #loginPage #loginButtonUp {
184 | cursor: pointer;
185 | font-weight: 600;
186 | }
187 |
188 | #loginPage {
189 | // display: flex;
190 | // direction: row;
191 | border: 2px solid grey;
192 | border-radius: 10px;
193 | width: auto;
194 | background-color:#fffafa;
195 | align-items: center;
196 | }
197 |
198 | .logDate {
199 | grid-area: date;
200 | }
201 | .logType {
202 | grid-area: type;
203 | }
204 |
205 | #password {
206 | -webkit-text-security: disc;
207 | // text-security: disc;
208 | }
209 | .podName {
210 | grid-area: pod;
211 | }
212 | .logMessage {
213 | grid-area: message;
214 | }
215 |
216 | #loginLogo {
217 | // background-color: red;
218 | align-items: center;
219 | // width: 100%;
220 | }
221 |
222 | .logRowComponent {
223 | background-color: #FAF9F9;
224 | color: black;
225 | margin: .5em;
226 | padding: .25em;
227 | display: grid;
228 | grid-template: 'date type type pod pod message message message'
229 | }
230 |
231 | .primary-btn {
232 | background-color: #1D3557;
233 | color: #F1FAEE
234 | }
235 | .secondary-btn {
236 | background-color: #FAF9F9;
237 |
238 | }
239 |
240 | .counter {
241 | background-color: #576e8f;
242 | }
243 |
244 | .gauge {
245 | background-color:cadetblue;
246 | }
247 |
248 | .histogram {
249 | background-color: rgb(180, 98, 114);
250 | }
251 | // #zingchart-react-0{
252 | // height: 500px;
253 | // }
254 | .chart {
255 | display: flex;
256 | flex-direction: row;
257 | justify-content: space-evenly;
258 | width: 100%;
259 | height: 100%;
260 | min-height: 200px;
261 | margin: 0px;
262 | padding: 0px;
263 | }
--------------------------------------------------------------------------------
/public/client/Containers/LiveQueryContainer.jsx:
--------------------------------------------------------------------------------
1 | import React, {useEffect, useState} from 'react';
2 | import { connect } from 'react-redux';
3 | import * as actions from '../actions/logsActionCreator.js';
4 | import Autocomplete from '@mui/material/Autocomplete';
5 | import TextField from '@mui/material/TextField';
6 | import Button from '@mui/material/Button';
7 | import { makeStyles } from '@mui/styles';
8 |
9 | const mapStateToProps = (state) => {
10 | return {
11 | appLogFields: state.logsReducer.appLogFields,
12 | selectedIndex: state.logsReducer.selectedIndex
13 | }
14 | }
15 |
16 | const mapDispatchToProps = (dispatch) => {
17 |
18 | return {
19 | getAppLogFields: () => dispatch(actions.getAppLogFields()),
20 | getAppLogs: (obj) => dispatch(actions.getAppLogs(obj)),
21 | selectIndex: (index) => dispatch(actions.selectIndex(index))
22 | }
23 | }
24 |
25 | //sample data;; store it with property named label;
26 | const testQueryIndex = [
27 | {label: "loggen-app"},
28 | {label: "fluent-d1231"}
29 | ]
30 |
31 | const testQueryField = [
32 | {label: "type"},
33 | {label: "podName"}
34 | ]
35 |
36 |
37 | // const QueryContainer = ({isPersist}) => {
38 | // const display = [];
39 | //depending on the current tab, rendered components will be different
40 | // if(isPersist){
41 |
42 | // }
43 |
44 | const LiveQueryContainer = (props) => {
45 |
46 |
47 | return (
48 |
49 |
}
55 | />
56 | }
62 | />
63 |
70 | console.log('add some function')}>
75 | Search
76 |
77 |
78 | );
79 | }
80 | // class QueryContainer extends React.Component {
81 | // constructor(props) {
82 | // super(props);
83 | // this.queryObj = {'all':false,'value':'','field':'','name':this.props.appLogFields[this.props.selectedIndex]}
84 | // this.sendQuery = this.sendQuery.bind(this);
85 | // this.addValue = this.addValue.bind(this);
86 | // this.selectField = this.selectField.bind(this);
87 | // this.selectIndex = this.selectIndex.bind(this);
88 | // this.indices = [];
89 | // this.fields = [];
90 | // this.getFields = this.getFields.bind(this);
91 | // }
92 | // getFields(){
93 | // console.log("hello from query container")
94 | // this.props.getAppLogFields();
95 | // console.log('appLogFields in cdm',this.props.appLogFields);
96 | // if(this.props.appLogFields.length){
97 | // let indexCounter = -1
98 | // this.indices = this.props.appLogFields.map((obj)=>{
99 | // indexCounter++;
100 | // return
`${obj[indexCounter]}`
101 | // })
102 | // console.log("indices",this.indices)
103 | // let fieldsCounter = -1;
104 | // this.fields = this.props.appLogFields[this.props.selectedIndex].fields.map((obj)=>{
105 | // fieldsCounter++;
106 | // return
`${obj[fieldsCounter]}`
107 | // })
108 | // console.log("fields",this.fields)
109 | // }
110 | // }
111 | // componentDidMount() {
112 | // //setInterval(this.getFields,1000)
113 |
114 | // // console.log('props after fetched in Component did mount', this.props);
115 | // }
116 | // sendQuery(){
117 | // if(this.queryObj.value) this.props.getAppLogs();
118 | // else {
119 | // this.queryObj.all = true;
120 | // this.props.getAppLogs(this.queryObj);
121 | // }
122 | // }
123 | // selectIndex(e){
124 | // console.log("index",e.value)
125 | // this.props.selectIndex(e.value);
126 | // }
127 | // selectField(e){
128 | // console.log("field",e.value)
129 | // this.queryObj.field = e.value;
130 | // }
131 | // addValue(e){
132 | // console.log("value",e.value)
133 | // this.queryObj.value = e.value;
134 | // }
135 |
136 |
137 |
138 | // render(){
139 |
140 | // return (
141 | //
142 | //
}
152 | // />
153 | // }
160 | // />
161 | //
162 | // Search
163 | //
164 | // );
165 | // }
166 |
167 | // };
168 |
169 | /*
170 |
Choose Index Name:
171 | {
172 | {this.indices}
173 | }
174 |
175 |
Choose Field Name:
176 |
177 | {this.fields}
178 |
179 |
180 |
181 | //
182 | //
Submit Query
183 | //
Reset Query
184 | // filter menu component
185 | */
186 |
187 |
188 |
189 |
190 | export default connect(mapStateToProps, mapDispatchToProps)(LiveQueryContainer);
--------------------------------------------------------------------------------
/server/controllers/metricsController.js:
--------------------------------------------------------------------------------
1 | const axios = require('axios');
2 | const metricsController = {};
3 |
4 | //to predefine and provide current, previous and step to get arrays of values to display data maybe make it customizable?
5 | //current date
6 | const endDate = new Date();
7 | //one hour as initial for start
8 | let startSet = 1;
9 | let dayStartSet = 24;
10 | //an hour before current
11 | const startDate = new Date(endDate.getTime()-startSet*3600000);
12 | const dayStartDate = new Date(endDate.getTime()-dayStartSet*3600000);
13 | //step initial
14 | let step = '1m';
15 | let dayStep = '24m';
16 |
17 |
18 | //top 4 relevant metrics by each node in the cluster
19 | //CPU saturation % by the node
20 | metricsController.getCPUSatByNodes = (req, res, next) => {
21 | res.locals.nodeMetrics = {};
22 | axios.get(`http://localhost:9090/api/v1/query_range?query=(sum(node_load15)%20by%20(instance)%20/%20count(node_cpu_seconds_total%7Bmode="system"%7D)%20by%20(instance))*100&start=${dayStartDate.toISOString()}&end=${endDate.toISOString()}&step=${dayStep}`)
23 | .then(data => {
24 | res.locals.nodeMetrics.CPUSatValsNodes = data.data.data.result;
25 | next();
26 | })
27 | .catch(err=>next(err));
28 | };
29 |
30 | //CPU utilization % by the node
31 | metricsController.getCPUByNodes = (req, res, next) => {
32 | axios.get(`http://localhost:9090/api/v1/query_range?query=100%20-%20(avg%20by%20(instance)%20(irate(node_cpu_seconds_total{mode=%22idle%22}[60m]))%20*%20100)&start=${startDate.toISOString()}&end=${endDate.toISOString()}&step=${step}`)
33 | .then(data => {
34 | res.locals.nodeMetrics.CPUNodes = data.data.data.result;
35 | next();
36 | })
37 | .catch(err=>next(err));
38 | };
39 |
40 | //Memory utilization % by the node
41 | metricsController.getMemoryByNodes = (req, res, next) => {
42 | axios.get(`http://localhost:9090/api/v1/query_range?query=sum((1-(node_memory_MemAvailable_bytes/node_memory_MemTotal_bytes))*100)%20by%20(instance)&start=${startDate.toISOString()}&end=${endDate.toISOString()}&step=${step}`)
43 | .then(data => {
44 | res.locals.nodeMetrics.MemoryNodes = data.data.data.result;
45 | next();
46 | })
47 | .catch(err=>next(err));
48 | };
49 |
50 | //WriteToDisk rate by the node
51 | metricsController.getWriteToDiskRateByNodes = (req, res, next) => {
52 | axios.get(`http://localhost:9090/api/v1/query_range?query=sum(rate(node_disk_written_bytes_total[60m]))by(instance)&start=${dayStartDate.toISOString()}&end=${endDate.toISOString()}&step=${dayStep}`)
53 | .then(data => {
54 | res.locals.nodeMetrics.WriteToDiskNodes = data.data.data.result;
55 | next();
56 | })
57 | .catch(err=>next(err));
58 | }
59 |
60 | //For Horizontal memory bar graph (to be developed) to present the memory allocation/availablity in the cluster
61 | metricsController.getMemoryBarData = (req, res, next) => {
62 | axios.get(`http://localhost:9090/api/v1/query?query=sum(cluster:namespace:pod_memory:active:kube_pod_container_resource_limits) by (node) / sum(machine_memory_bytes) by (node)`)
63 | .then(data => {
64 | res.locals.nodeMetrics.MemoryBarGraph = data.data.data.result;
65 | next();
66 | })
67 | .catch(err=>next(err));
68 | }
69 |
70 | //pod metrics: node's name will be added as reqeust parameter, it will pull relevent pod metrics inside the node
71 |
72 | //cpu usage rate by pod inside one node
73 | metricsController.getCPUByPods = (req, res, next) => {
74 | res.locals.podMetrics = {};
75 | const node = req.params.nodeName;
76 | axios.get(`http://localhost:9090/api/v1/query_range?query=sum(rate(container_cpu_usage_seconds_total{node="${node}",pod!=%22POD%22,%20pod!=%22%22}[60m]))%20by%20(pod)&start=${startDate.toISOString()}&end=${endDate.toISOString()}&step=${step}`)
77 | .then(data => {
78 | res.locals.podMetrics.CPUPods = data.data.data.result;
79 | next();
80 | })
81 | .catch(err=>next(err));
82 | };
83 |
84 | //memory usuage by pod inside one node
85 | metricsController.getMemoryByPods = (req, res, next) => {
86 | const node = req.params.nodeName;
87 | axios.get(`http://localhost:9090/api/v1/query_range?query=sum(container_memory_working_set_bytes{node="${node}",pod!=%22POD%22,%20pod!=%22%22})%20by%20(pod)&start=${startDate.toISOString()}&end=${endDate.toISOString()}&step=${step}`)
88 | .then(data => {
89 | res.locals.podMetrics.MemoryPods = data.data.data.result;
90 | next();
91 | })
92 | .catch(err=>next(err));
93 | };
94 |
95 | //disk write rate by pod inside one node
96 | metricsController.getWriteToDiskRateByPods = (req, res, next) => {
97 | const node = req.params.nodeName;
98 | axios.get(`http://localhost:9090/api/v1/query_range?query=sum(rate(container_fs_writes_bytes_total{node="${node}",pod!=%22POD%22,%20pod!=%22%22}[60m]))%20by%20(pod)&start=${dayStartDate.toISOString()}&end=${endDate.toISOString()}&step=${dayStep}`)
99 | .then(data => {
100 | res.locals.podMetrics.WriteToDiskPods = data.data.data.result;
101 | next();
102 | })
103 | .catch(err=>next(err));
104 | }
105 |
106 | //kubelet logs by pod inside one node
107 | metricsController.getLogsByPods = (req, res, next) => {
108 | const node = req.params.nodeName;
109 | axios.get(`http://localhost:9090/api/v1/query_range?query=sum(kubelet_container_log_filesystem_used_bytes{node="${node}",pod!=%22POD%22,%20pod!=%22%22})%20by%20(pod)&start=${dayStartDate.toISOString()}&end=${endDate.toISOString()}&step=${dayStep}`)
110 | .then(data => {
111 | res.locals.podMetrics.LogsByPods = data.data.data.result;
112 | next();
113 | })
114 | .catch(err=>next(err));
115 | }
116 |
117 | //use promise all to resolve multiple axios requests to pull relevant control plane/master node components
118 | metricsController.getMasterNodeMetrics = (req, res, next) => {
119 | const urls = {
120 | serverAPILatency: `http://localhost:9090/api/v1/query_range?query=sum(cluster_quantile:apiserver_request_duration_seconds:histogram_quantile{resource!="",quantile="0.9"}) by (resource)&start=${startDate.toISOString()}&end=${endDate.toISOString()}&step=${step}`,
121 | serverAPIsuccessReq: `http://localhost:9090/api/v1/query_range?query=sum(rate(apiserver_request_total{job="apiserver",code=~"2..",group!=""}[60m])) by (group)&start=${startDate.toISOString()}&end=${endDate.toISOString()}&step=${step}`,
122 | controllerAddCounter: `http://localhost:9090/api/v1/query_range?query=rate(workqueue_adds_total[60m])&start=${startDate.toISOString()}&end=${endDate.toISOString()}&step=${step}`,
123 | etcdRequestRate: `http://localhost:9090/api/v1/query_range?query=sum(rate(etcd_request_duration_seconds_sum[60m]))by(operation)&start=${startDate.toISOString()}&end=${endDate.toISOString()}&step=${step}`,
124 | }
125 |
126 | const promises = [];
127 |
128 | for(let url in urls){
129 | promises.push(axios.get(urls[url]));
130 | };
131 |
132 | Promise.all(promises)
133 | .then(results => {
134 | let index = 0;
135 | for(let url in urls){
136 | urls[url] = results[index].data.data.result;
137 | index++;
138 | }
139 | res.locals.masterNode = urls;
140 | next();
141 | })
142 | .catch(err=>next(err));
143 | };
144 |
145 | module.exports = metricsController;
146 |
147 |
--------------------------------------------------------------------------------
/logGen-app/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "logGen-app",
3 | "version": "1.0.0",
4 | "lockfileVersion": 2,
5 | "requires": true,
6 | "packages": {
7 | "": {
8 | "name": "logGen-app",
9 | "version": "1.0.0",
10 | "license": "ISC",
11 | "dependencies": {
12 | "express": "^4.17.1",
13 | "node-fetch": "^3.0.0",
14 | "pino": "6.3.1"
15 | }
16 | },
17 | "node_modules/accepts": {
18 | "version": "1.3.7",
19 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
20 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
21 | "dependencies": {
22 | "mime-types": "~2.1.24",
23 | "negotiator": "0.6.2"
24 | },
25 | "engines": {
26 | "node": ">= 0.6"
27 | }
28 | },
29 | "node_modules/array-flatten": {
30 | "version": "1.1.1",
31 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
32 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
33 | },
34 | "node_modules/atomic-sleep": {
35 | "version": "1.0.0",
36 | "resolved": "https://registry.npmjs.org/atomic-sleep/-/atomic-sleep-1.0.0.tgz",
37 | "integrity": "sha512-kNOjDqAh7px0XWNI+4QbzoiR/nTkHAWNud2uvnJquD1/x5a7EQZMJT0AczqK0Qn67oY/TTQ1LbUKajZpp3I9tQ==",
38 | "engines": {
39 | "node": ">=8.0.0"
40 | }
41 | },
42 | "node_modules/body-parser": {
43 | "version": "1.19.0",
44 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
45 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
46 | "dependencies": {
47 | "bytes": "3.1.0",
48 | "content-type": "~1.0.4",
49 | "debug": "2.6.9",
50 | "depd": "~1.1.2",
51 | "http-errors": "1.7.2",
52 | "iconv-lite": "0.4.24",
53 | "on-finished": "~2.3.0",
54 | "qs": "6.7.0",
55 | "raw-body": "2.4.0",
56 | "type-is": "~1.6.17"
57 | },
58 | "engines": {
59 | "node": ">= 0.8"
60 | }
61 | },
62 | "node_modules/bytes": {
63 | "version": "3.1.0",
64 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
65 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==",
66 | "engines": {
67 | "node": ">= 0.8"
68 | }
69 | },
70 | "node_modules/content-disposition": {
71 | "version": "0.5.3",
72 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
73 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
74 | "dependencies": {
75 | "safe-buffer": "5.1.2"
76 | },
77 | "engines": {
78 | "node": ">= 0.6"
79 | }
80 | },
81 | "node_modules/content-type": {
82 | "version": "1.0.4",
83 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
84 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==",
85 | "engines": {
86 | "node": ">= 0.6"
87 | }
88 | },
89 | "node_modules/cookie": {
90 | "version": "0.4.0",
91 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
92 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==",
93 | "engines": {
94 | "node": ">= 0.6"
95 | }
96 | },
97 | "node_modules/cookie-signature": {
98 | "version": "1.0.6",
99 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
100 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
101 | },
102 | "node_modules/data-uri-to-buffer": {
103 | "version": "3.0.1",
104 | "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-3.0.1.tgz",
105 | "integrity": "sha512-WboRycPNsVw3B3TL559F7kuBUM4d8CgMEvk6xEJlOp7OBPjt6G7z8WMWlD2rOFZLk6OYfFIUGsCOWzcQH9K2og==",
106 | "engines": {
107 | "node": ">= 6"
108 | }
109 | },
110 | "node_modules/debug": {
111 | "version": "2.6.9",
112 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
113 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
114 | "dependencies": {
115 | "ms": "2.0.0"
116 | }
117 | },
118 | "node_modules/depd": {
119 | "version": "1.1.2",
120 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
121 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=",
122 | "engines": {
123 | "node": ">= 0.6"
124 | }
125 | },
126 | "node_modules/destroy": {
127 | "version": "1.0.4",
128 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
129 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
130 | },
131 | "node_modules/ee-first": {
132 | "version": "1.1.1",
133 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
134 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
135 | },
136 | "node_modules/encodeurl": {
137 | "version": "1.0.2",
138 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
139 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=",
140 | "engines": {
141 | "node": ">= 0.8"
142 | }
143 | },
144 | "node_modules/escape-html": {
145 | "version": "1.0.3",
146 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
147 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
148 | },
149 | "node_modules/etag": {
150 | "version": "1.8.1",
151 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
152 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=",
153 | "engines": {
154 | "node": ">= 0.6"
155 | }
156 | },
157 | "node_modules/express": {
158 | "version": "4.17.1",
159 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
160 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
161 | "dependencies": {
162 | "accepts": "~1.3.7",
163 | "array-flatten": "1.1.1",
164 | "body-parser": "1.19.0",
165 | "content-disposition": "0.5.3",
166 | "content-type": "~1.0.4",
167 | "cookie": "0.4.0",
168 | "cookie-signature": "1.0.6",
169 | "debug": "2.6.9",
170 | "depd": "~1.1.2",
171 | "encodeurl": "~1.0.2",
172 | "escape-html": "~1.0.3",
173 | "etag": "~1.8.1",
174 | "finalhandler": "~1.1.2",
175 | "fresh": "0.5.2",
176 | "merge-descriptors": "1.0.1",
177 | "methods": "~1.1.2",
178 | "on-finished": "~2.3.0",
179 | "parseurl": "~1.3.3",
180 | "path-to-regexp": "0.1.7",
181 | "proxy-addr": "~2.0.5",
182 | "qs": "6.7.0",
183 | "range-parser": "~1.2.1",
184 | "safe-buffer": "5.1.2",
185 | "send": "0.17.1",
186 | "serve-static": "1.14.1",
187 | "setprototypeof": "1.1.1",
188 | "statuses": "~1.5.0",
189 | "type-is": "~1.6.18",
190 | "utils-merge": "1.0.1",
191 | "vary": "~1.1.2"
192 | },
193 | "engines": {
194 | "node": ">= 0.10.0"
195 | }
196 | },
197 | "node_modules/fast-redact": {
198 | "version": "2.1.0",
199 | "resolved": "https://registry.npmjs.org/fast-redact/-/fast-redact-2.1.0.tgz",
200 | "integrity": "sha512-0LkHpTLyadJavq9sRzzyqIoMZemWli77K2/MGOkafrR64B9ItrvZ9aT+jluvNDsv0YEHjSNhlMBtbokuoqii4A==",
201 | "engines": {
202 | "node": ">=6"
203 | }
204 | },
205 | "node_modules/fast-safe-stringify": {
206 | "version": "2.1.1",
207 | "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz",
208 | "integrity": "sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA=="
209 | },
210 | "node_modules/fetch-blob": {
211 | "version": "3.1.2",
212 | "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.1.2.tgz",
213 | "integrity": "sha512-hunJbvy/6OLjCD0uuhLdp0mMPzP/yd2ssd1t2FCJsaA7wkWhpbp9xfuNVpv7Ll4jFhzp6T4LAupSiV9uOeg0VQ==",
214 | "funding": [
215 | {
216 | "type": "github",
217 | "url": "https://github.com/sponsors/jimmywarting"
218 | },
219 | {
220 | "type": "paypal",
221 | "url": "https://paypal.me/jimmywarting"
222 | }
223 | ],
224 | "dependencies": {
225 | "web-streams-polyfill": "^3.0.3"
226 | },
227 | "engines": {
228 | "node": "^12.20 || >= 14.13"
229 | }
230 | },
231 | "node_modules/finalhandler": {
232 | "version": "1.1.2",
233 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
234 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
235 | "dependencies": {
236 | "debug": "2.6.9",
237 | "encodeurl": "~1.0.2",
238 | "escape-html": "~1.0.3",
239 | "on-finished": "~2.3.0",
240 | "parseurl": "~1.3.3",
241 | "statuses": "~1.5.0",
242 | "unpipe": "~1.0.0"
243 | },
244 | "engines": {
245 | "node": ">= 0.8"
246 | }
247 | },
248 | "node_modules/flatstr": {
249 | "version": "1.0.12",
250 | "resolved": "https://registry.npmjs.org/flatstr/-/flatstr-1.0.12.tgz",
251 | "integrity": "sha512-4zPxDyhCyiN2wIAtSLI6gc82/EjqZc1onI4Mz/l0pWrAlsSfYH/2ZIcU+e3oA2wDwbzIWNKwa23F8rh6+DRWkw=="
252 | },
253 | "node_modules/forwarded": {
254 | "version": "0.2.0",
255 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
256 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
257 | "engines": {
258 | "node": ">= 0.6"
259 | }
260 | },
261 | "node_modules/fresh": {
262 | "version": "0.5.2",
263 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
264 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=",
265 | "engines": {
266 | "node": ">= 0.6"
267 | }
268 | },
269 | "node_modules/http-errors": {
270 | "version": "1.7.2",
271 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
272 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
273 | "dependencies": {
274 | "depd": "~1.1.2",
275 | "inherits": "2.0.3",
276 | "setprototypeof": "1.1.1",
277 | "statuses": ">= 1.5.0 < 2",
278 | "toidentifier": "1.0.0"
279 | },
280 | "engines": {
281 | "node": ">= 0.6"
282 | }
283 | },
284 | "node_modules/iconv-lite": {
285 | "version": "0.4.24",
286 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
287 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
288 | "dependencies": {
289 | "safer-buffer": ">= 2.1.2 < 3"
290 | },
291 | "engines": {
292 | "node": ">=0.10.0"
293 | }
294 | },
295 | "node_modules/inherits": {
296 | "version": "2.0.3",
297 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
298 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
299 | },
300 | "node_modules/ipaddr.js": {
301 | "version": "1.9.1",
302 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
303 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
304 | "engines": {
305 | "node": ">= 0.10"
306 | }
307 | },
308 | "node_modules/media-typer": {
309 | "version": "0.3.0",
310 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
311 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=",
312 | "engines": {
313 | "node": ">= 0.6"
314 | }
315 | },
316 | "node_modules/merge-descriptors": {
317 | "version": "1.0.1",
318 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
319 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
320 | },
321 | "node_modules/methods": {
322 | "version": "1.1.2",
323 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
324 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=",
325 | "engines": {
326 | "node": ">= 0.6"
327 | }
328 | },
329 | "node_modules/mime": {
330 | "version": "1.6.0",
331 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
332 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==",
333 | "bin": {
334 | "mime": "cli.js"
335 | },
336 | "engines": {
337 | "node": ">=4"
338 | }
339 | },
340 | "node_modules/mime-db": {
341 | "version": "1.50.0",
342 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.50.0.tgz",
343 | "integrity": "sha512-9tMZCDlYHqeERXEHO9f/hKfNXhre5dK2eE/krIvUjZbS2KPcqGDfNShIWS1uW9XOTKQKqK6qbeOci18rbfW77A==",
344 | "engines": {
345 | "node": ">= 0.6"
346 | }
347 | },
348 | "node_modules/mime-types": {
349 | "version": "2.1.33",
350 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.33.tgz",
351 | "integrity": "sha512-plLElXp7pRDd0bNZHw+nMd52vRYjLwQjygaNg7ddJ2uJtTlmnTCjWuPKxVu6//AdaRuME84SvLW91sIkBqGT0g==",
352 | "dependencies": {
353 | "mime-db": "1.50.0"
354 | },
355 | "engines": {
356 | "node": ">= 0.6"
357 | }
358 | },
359 | "node_modules/ms": {
360 | "version": "2.0.0",
361 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
362 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
363 | },
364 | "node_modules/negotiator": {
365 | "version": "0.6.2",
366 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
367 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==",
368 | "engines": {
369 | "node": ">= 0.6"
370 | }
371 | },
372 | "node_modules/node-fetch": {
373 | "version": "3.0.0",
374 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.0.0.tgz",
375 | "integrity": "sha512-bKMI+C7/T/SPU1lKnbQbwxptpCrG9ashG+VkytmXCPZyuM9jB6VU+hY0oi4lC8LxTtAeWdckNCTa3nrGsAdA3Q==",
376 | "dependencies": {
377 | "data-uri-to-buffer": "^3.0.1",
378 | "fetch-blob": "^3.1.2"
379 | },
380 | "engines": {
381 | "node": "^12.20.0 || ^14.13.1 || >=16.0.0"
382 | },
383 | "funding": {
384 | "type": "opencollective",
385 | "url": "https://opencollective.com/node-fetch"
386 | }
387 | },
388 | "node_modules/on-finished": {
389 | "version": "2.3.0",
390 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
391 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
392 | "dependencies": {
393 | "ee-first": "1.1.1"
394 | },
395 | "engines": {
396 | "node": ">= 0.8"
397 | }
398 | },
399 | "node_modules/parseurl": {
400 | "version": "1.3.3",
401 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
402 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
403 | "engines": {
404 | "node": ">= 0.8"
405 | }
406 | },
407 | "node_modules/path-to-regexp": {
408 | "version": "0.1.7",
409 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
410 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
411 | },
412 | "node_modules/pino": {
413 | "version": "6.3.1",
414 | "resolved": "https://registry.npmjs.org/pino/-/pino-6.3.1.tgz",
415 | "integrity": "sha512-RgT010a5FfnxJ2AwB0TqcEuM+gNsnd08PZnCob98JSTLldLF0GMFJ/Z1VE/rdl5yJCqcoLwftmZSwSFY4/Hc2g==",
416 | "dependencies": {
417 | "fast-redact": "^2.0.0",
418 | "fast-safe-stringify": "^2.0.7",
419 | "flatstr": "^1.0.12",
420 | "pino-std-serializers": "^2.4.2",
421 | "quick-format-unescaped": "^4.0.1",
422 | "sonic-boom": "^1.0.0"
423 | },
424 | "bin": {
425 | "pino": "bin.js"
426 | }
427 | },
428 | "node_modules/pino-std-serializers": {
429 | "version": "2.5.0",
430 | "resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-2.5.0.tgz",
431 | "integrity": "sha512-wXqbqSrIhE58TdrxxlfLwU9eDhrzppQDvGhBEr1gYbzzM4KKo3Y63gSjiDXRKLVS2UOXdPNR2v+KnQgNrs+xUg=="
432 | },
433 | "node_modules/proxy-addr": {
434 | "version": "2.0.7",
435 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
436 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
437 | "dependencies": {
438 | "forwarded": "0.2.0",
439 | "ipaddr.js": "1.9.1"
440 | },
441 | "engines": {
442 | "node": ">= 0.10"
443 | }
444 | },
445 | "node_modules/qs": {
446 | "version": "6.7.0",
447 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
448 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==",
449 | "engines": {
450 | "node": ">=0.6"
451 | }
452 | },
453 | "node_modules/quick-format-unescaped": {
454 | "version": "4.0.4",
455 | "resolved": "https://registry.npmjs.org/quick-format-unescaped/-/quick-format-unescaped-4.0.4.tgz",
456 | "integrity": "sha512-tYC1Q1hgyRuHgloV/YXs2w15unPVh8qfu/qCTfhTYamaw7fyhumKa2yGpdSo87vY32rIclj+4fWYQXUMs9EHvg=="
457 | },
458 | "node_modules/range-parser": {
459 | "version": "1.2.1",
460 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
461 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
462 | "engines": {
463 | "node": ">= 0.6"
464 | }
465 | },
466 | "node_modules/raw-body": {
467 | "version": "2.4.0",
468 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
469 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
470 | "dependencies": {
471 | "bytes": "3.1.0",
472 | "http-errors": "1.7.2",
473 | "iconv-lite": "0.4.24",
474 | "unpipe": "1.0.0"
475 | },
476 | "engines": {
477 | "node": ">= 0.8"
478 | }
479 | },
480 | "node_modules/safe-buffer": {
481 | "version": "5.1.2",
482 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
483 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
484 | },
485 | "node_modules/safer-buffer": {
486 | "version": "2.1.2",
487 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
488 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
489 | },
490 | "node_modules/send": {
491 | "version": "0.17.1",
492 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
493 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
494 | "dependencies": {
495 | "debug": "2.6.9",
496 | "depd": "~1.1.2",
497 | "destroy": "~1.0.4",
498 | "encodeurl": "~1.0.2",
499 | "escape-html": "~1.0.3",
500 | "etag": "~1.8.1",
501 | "fresh": "0.5.2",
502 | "http-errors": "~1.7.2",
503 | "mime": "1.6.0",
504 | "ms": "2.1.1",
505 | "on-finished": "~2.3.0",
506 | "range-parser": "~1.2.1",
507 | "statuses": "~1.5.0"
508 | },
509 | "engines": {
510 | "node": ">= 0.8.0"
511 | }
512 | },
513 | "node_modules/send/node_modules/ms": {
514 | "version": "2.1.1",
515 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
516 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
517 | },
518 | "node_modules/serve-static": {
519 | "version": "1.14.1",
520 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
521 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
522 | "dependencies": {
523 | "encodeurl": "~1.0.2",
524 | "escape-html": "~1.0.3",
525 | "parseurl": "~1.3.3",
526 | "send": "0.17.1"
527 | },
528 | "engines": {
529 | "node": ">= 0.8.0"
530 | }
531 | },
532 | "node_modules/setprototypeof": {
533 | "version": "1.1.1",
534 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
535 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
536 | },
537 | "node_modules/sonic-boom": {
538 | "version": "1.4.1",
539 | "resolved": "https://registry.npmjs.org/sonic-boom/-/sonic-boom-1.4.1.tgz",
540 | "integrity": "sha512-LRHh/A8tpW7ru89lrlkU4AszXt1dbwSjVWguGrmlxE7tawVmDBlI1PILMkXAxJTwqhgsEeTHzj36D5CmHgQmNg==",
541 | "dependencies": {
542 | "atomic-sleep": "^1.0.0",
543 | "flatstr": "^1.0.12"
544 | }
545 | },
546 | "node_modules/statuses": {
547 | "version": "1.5.0",
548 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
549 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=",
550 | "engines": {
551 | "node": ">= 0.6"
552 | }
553 | },
554 | "node_modules/toidentifier": {
555 | "version": "1.0.0",
556 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
557 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==",
558 | "engines": {
559 | "node": ">=0.6"
560 | }
561 | },
562 | "node_modules/type-is": {
563 | "version": "1.6.18",
564 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
565 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
566 | "dependencies": {
567 | "media-typer": "0.3.0",
568 | "mime-types": "~2.1.24"
569 | },
570 | "engines": {
571 | "node": ">= 0.6"
572 | }
573 | },
574 | "node_modules/unpipe": {
575 | "version": "1.0.0",
576 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
577 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=",
578 | "engines": {
579 | "node": ">= 0.8"
580 | }
581 | },
582 | "node_modules/utils-merge": {
583 | "version": "1.0.1",
584 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
585 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=",
586 | "engines": {
587 | "node": ">= 0.4.0"
588 | }
589 | },
590 | "node_modules/vary": {
591 | "version": "1.1.2",
592 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
593 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=",
594 | "engines": {
595 | "node": ">= 0.8"
596 | }
597 | },
598 | "node_modules/web-streams-polyfill": {
599 | "version": "3.1.1",
600 | "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.1.1.tgz",
601 | "integrity": "sha512-Czi3fG883e96T4DLEPRvufrF2ydhOOW1+1a6c3gNjH2aIh50DNFBdfwh2AKoOf1rXvpvavAoA11Qdq9+BKjE0Q==",
602 | "engines": {
603 | "node": ">= 8"
604 | }
605 | }
606 | },
607 | "dependencies": {
608 | "accepts": {
609 | "version": "1.3.7",
610 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz",
611 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==",
612 | "requires": {
613 | "mime-types": "~2.1.24",
614 | "negotiator": "0.6.2"
615 | }
616 | },
617 | "array-flatten": {
618 | "version": "1.1.1",
619 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
620 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI="
621 | },
622 | "atomic-sleep": {
623 | "version": "1.0.0",
624 | "resolved": "https://registry.npmjs.org/atomic-sleep/-/atomic-sleep-1.0.0.tgz",
625 | "integrity": "sha512-kNOjDqAh7px0XWNI+4QbzoiR/nTkHAWNud2uvnJquD1/x5a7EQZMJT0AczqK0Qn67oY/TTQ1LbUKajZpp3I9tQ=="
626 | },
627 | "body-parser": {
628 | "version": "1.19.0",
629 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz",
630 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==",
631 | "requires": {
632 | "bytes": "3.1.0",
633 | "content-type": "~1.0.4",
634 | "debug": "2.6.9",
635 | "depd": "~1.1.2",
636 | "http-errors": "1.7.2",
637 | "iconv-lite": "0.4.24",
638 | "on-finished": "~2.3.0",
639 | "qs": "6.7.0",
640 | "raw-body": "2.4.0",
641 | "type-is": "~1.6.17"
642 | }
643 | },
644 | "bytes": {
645 | "version": "3.1.0",
646 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz",
647 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg=="
648 | },
649 | "content-disposition": {
650 | "version": "0.5.3",
651 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz",
652 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==",
653 | "requires": {
654 | "safe-buffer": "5.1.2"
655 | }
656 | },
657 | "content-type": {
658 | "version": "1.0.4",
659 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz",
660 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="
661 | },
662 | "cookie": {
663 | "version": "0.4.0",
664 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz",
665 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg=="
666 | },
667 | "cookie-signature": {
668 | "version": "1.0.6",
669 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
670 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw="
671 | },
672 | "data-uri-to-buffer": {
673 | "version": "3.0.1",
674 | "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-3.0.1.tgz",
675 | "integrity": "sha512-WboRycPNsVw3B3TL559F7kuBUM4d8CgMEvk6xEJlOp7OBPjt6G7z8WMWlD2rOFZLk6OYfFIUGsCOWzcQH9K2og=="
676 | },
677 | "debug": {
678 | "version": "2.6.9",
679 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
680 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
681 | "requires": {
682 | "ms": "2.0.0"
683 | }
684 | },
685 | "depd": {
686 | "version": "1.1.2",
687 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
688 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak="
689 | },
690 | "destroy": {
691 | "version": "1.0.4",
692 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
693 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA="
694 | },
695 | "ee-first": {
696 | "version": "1.1.1",
697 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
698 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
699 | },
700 | "encodeurl": {
701 | "version": "1.0.2",
702 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
703 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k="
704 | },
705 | "escape-html": {
706 | "version": "1.0.3",
707 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
708 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg="
709 | },
710 | "etag": {
711 | "version": "1.8.1",
712 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
713 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc="
714 | },
715 | "express": {
716 | "version": "4.17.1",
717 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz",
718 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==",
719 | "requires": {
720 | "accepts": "~1.3.7",
721 | "array-flatten": "1.1.1",
722 | "body-parser": "1.19.0",
723 | "content-disposition": "0.5.3",
724 | "content-type": "~1.0.4",
725 | "cookie": "0.4.0",
726 | "cookie-signature": "1.0.6",
727 | "debug": "2.6.9",
728 | "depd": "~1.1.2",
729 | "encodeurl": "~1.0.2",
730 | "escape-html": "~1.0.3",
731 | "etag": "~1.8.1",
732 | "finalhandler": "~1.1.2",
733 | "fresh": "0.5.2",
734 | "merge-descriptors": "1.0.1",
735 | "methods": "~1.1.2",
736 | "on-finished": "~2.3.0",
737 | "parseurl": "~1.3.3",
738 | "path-to-regexp": "0.1.7",
739 | "proxy-addr": "~2.0.5",
740 | "qs": "6.7.0",
741 | "range-parser": "~1.2.1",
742 | "safe-buffer": "5.1.2",
743 | "send": "0.17.1",
744 | "serve-static": "1.14.1",
745 | "setprototypeof": "1.1.1",
746 | "statuses": "~1.5.0",
747 | "type-is": "~1.6.18",
748 | "utils-merge": "1.0.1",
749 | "vary": "~1.1.2"
750 | }
751 | },
752 | "fast-redact": {
753 | "version": "2.1.0",
754 | "resolved": "https://registry.npmjs.org/fast-redact/-/fast-redact-2.1.0.tgz",
755 | "integrity": "sha512-0LkHpTLyadJavq9sRzzyqIoMZemWli77K2/MGOkafrR64B9ItrvZ9aT+jluvNDsv0YEHjSNhlMBtbokuoqii4A=="
756 | },
757 | "fast-safe-stringify": {
758 | "version": "2.1.1",
759 | "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz",
760 | "integrity": "sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA=="
761 | },
762 | "fetch-blob": {
763 | "version": "3.1.2",
764 | "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.1.2.tgz",
765 | "integrity": "sha512-hunJbvy/6OLjCD0uuhLdp0mMPzP/yd2ssd1t2FCJsaA7wkWhpbp9xfuNVpv7Ll4jFhzp6T4LAupSiV9uOeg0VQ==",
766 | "requires": {
767 | "web-streams-polyfill": "^3.0.3"
768 | }
769 | },
770 | "finalhandler": {
771 | "version": "1.1.2",
772 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz",
773 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==",
774 | "requires": {
775 | "debug": "2.6.9",
776 | "encodeurl": "~1.0.2",
777 | "escape-html": "~1.0.3",
778 | "on-finished": "~2.3.0",
779 | "parseurl": "~1.3.3",
780 | "statuses": "~1.5.0",
781 | "unpipe": "~1.0.0"
782 | }
783 | },
784 | "flatstr": {
785 | "version": "1.0.12",
786 | "resolved": "https://registry.npmjs.org/flatstr/-/flatstr-1.0.12.tgz",
787 | "integrity": "sha512-4zPxDyhCyiN2wIAtSLI6gc82/EjqZc1onI4Mz/l0pWrAlsSfYH/2ZIcU+e3oA2wDwbzIWNKwa23F8rh6+DRWkw=="
788 | },
789 | "forwarded": {
790 | "version": "0.2.0",
791 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
792 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow=="
793 | },
794 | "fresh": {
795 | "version": "0.5.2",
796 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
797 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac="
798 | },
799 | "http-errors": {
800 | "version": "1.7.2",
801 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz",
802 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==",
803 | "requires": {
804 | "depd": "~1.1.2",
805 | "inherits": "2.0.3",
806 | "setprototypeof": "1.1.1",
807 | "statuses": ">= 1.5.0 < 2",
808 | "toidentifier": "1.0.0"
809 | }
810 | },
811 | "iconv-lite": {
812 | "version": "0.4.24",
813 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
814 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
815 | "requires": {
816 | "safer-buffer": ">= 2.1.2 < 3"
817 | }
818 | },
819 | "inherits": {
820 | "version": "2.0.3",
821 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
822 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4="
823 | },
824 | "ipaddr.js": {
825 | "version": "1.9.1",
826 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
827 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="
828 | },
829 | "media-typer": {
830 | "version": "0.3.0",
831 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
832 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g="
833 | },
834 | "merge-descriptors": {
835 | "version": "1.0.1",
836 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
837 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E="
838 | },
839 | "methods": {
840 | "version": "1.1.2",
841 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
842 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4="
843 | },
844 | "mime": {
845 | "version": "1.6.0",
846 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz",
847 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg=="
848 | },
849 | "mime-db": {
850 | "version": "1.50.0",
851 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.50.0.tgz",
852 | "integrity": "sha512-9tMZCDlYHqeERXEHO9f/hKfNXhre5dK2eE/krIvUjZbS2KPcqGDfNShIWS1uW9XOTKQKqK6qbeOci18rbfW77A=="
853 | },
854 | "mime-types": {
855 | "version": "2.1.33",
856 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.33.tgz",
857 | "integrity": "sha512-plLElXp7pRDd0bNZHw+nMd52vRYjLwQjygaNg7ddJ2uJtTlmnTCjWuPKxVu6//AdaRuME84SvLW91sIkBqGT0g==",
858 | "requires": {
859 | "mime-db": "1.50.0"
860 | }
861 | },
862 | "ms": {
863 | "version": "2.0.0",
864 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
865 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g="
866 | },
867 | "negotiator": {
868 | "version": "0.6.2",
869 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz",
870 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw=="
871 | },
872 | "node-fetch": {
873 | "version": "3.0.0",
874 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.0.0.tgz",
875 | "integrity": "sha512-bKMI+C7/T/SPU1lKnbQbwxptpCrG9ashG+VkytmXCPZyuM9jB6VU+hY0oi4lC8LxTtAeWdckNCTa3nrGsAdA3Q==",
876 | "requires": {
877 | "data-uri-to-buffer": "^3.0.1",
878 | "fetch-blob": "^3.1.2"
879 | }
880 | },
881 | "on-finished": {
882 | "version": "2.3.0",
883 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
884 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=",
885 | "requires": {
886 | "ee-first": "1.1.1"
887 | }
888 | },
889 | "parseurl": {
890 | "version": "1.3.3",
891 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
892 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="
893 | },
894 | "path-to-regexp": {
895 | "version": "0.1.7",
896 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
897 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w="
898 | },
899 | "pino": {
900 | "version": "6.3.1",
901 | "resolved": "https://registry.npmjs.org/pino/-/pino-6.3.1.tgz",
902 | "integrity": "sha512-RgT010a5FfnxJ2AwB0TqcEuM+gNsnd08PZnCob98JSTLldLF0GMFJ/Z1VE/rdl5yJCqcoLwftmZSwSFY4/Hc2g==",
903 | "requires": {
904 | "fast-redact": "^2.0.0",
905 | "fast-safe-stringify": "^2.0.7",
906 | "flatstr": "^1.0.12",
907 | "pino-std-serializers": "^2.4.2",
908 | "quick-format-unescaped": "^4.0.1",
909 | "sonic-boom": "^1.0.0"
910 | }
911 | },
912 | "pino-std-serializers": {
913 | "version": "2.5.0",
914 | "resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-2.5.0.tgz",
915 | "integrity": "sha512-wXqbqSrIhE58TdrxxlfLwU9eDhrzppQDvGhBEr1gYbzzM4KKo3Y63gSjiDXRKLVS2UOXdPNR2v+KnQgNrs+xUg=="
916 | },
917 | "proxy-addr": {
918 | "version": "2.0.7",
919 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
920 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
921 | "requires": {
922 | "forwarded": "0.2.0",
923 | "ipaddr.js": "1.9.1"
924 | }
925 | },
926 | "qs": {
927 | "version": "6.7.0",
928 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz",
929 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ=="
930 | },
931 | "quick-format-unescaped": {
932 | "version": "4.0.4",
933 | "resolved": "https://registry.npmjs.org/quick-format-unescaped/-/quick-format-unescaped-4.0.4.tgz",
934 | "integrity": "sha512-tYC1Q1hgyRuHgloV/YXs2w15unPVh8qfu/qCTfhTYamaw7fyhumKa2yGpdSo87vY32rIclj+4fWYQXUMs9EHvg=="
935 | },
936 | "range-parser": {
937 | "version": "1.2.1",
938 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
939 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="
940 | },
941 | "raw-body": {
942 | "version": "2.4.0",
943 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz",
944 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==",
945 | "requires": {
946 | "bytes": "3.1.0",
947 | "http-errors": "1.7.2",
948 | "iconv-lite": "0.4.24",
949 | "unpipe": "1.0.0"
950 | }
951 | },
952 | "safe-buffer": {
953 | "version": "5.1.2",
954 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
955 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
956 | },
957 | "safer-buffer": {
958 | "version": "2.1.2",
959 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
960 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
961 | },
962 | "send": {
963 | "version": "0.17.1",
964 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz",
965 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==",
966 | "requires": {
967 | "debug": "2.6.9",
968 | "depd": "~1.1.2",
969 | "destroy": "~1.0.4",
970 | "encodeurl": "~1.0.2",
971 | "escape-html": "~1.0.3",
972 | "etag": "~1.8.1",
973 | "fresh": "0.5.2",
974 | "http-errors": "~1.7.2",
975 | "mime": "1.6.0",
976 | "ms": "2.1.1",
977 | "on-finished": "~2.3.0",
978 | "range-parser": "~1.2.1",
979 | "statuses": "~1.5.0"
980 | },
981 | "dependencies": {
982 | "ms": {
983 | "version": "2.1.1",
984 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
985 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg=="
986 | }
987 | }
988 | },
989 | "serve-static": {
990 | "version": "1.14.1",
991 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz",
992 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==",
993 | "requires": {
994 | "encodeurl": "~1.0.2",
995 | "escape-html": "~1.0.3",
996 | "parseurl": "~1.3.3",
997 | "send": "0.17.1"
998 | }
999 | },
1000 | "setprototypeof": {
1001 | "version": "1.1.1",
1002 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz",
1003 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw=="
1004 | },
1005 | "sonic-boom": {
1006 | "version": "1.4.1",
1007 | "resolved": "https://registry.npmjs.org/sonic-boom/-/sonic-boom-1.4.1.tgz",
1008 | "integrity": "sha512-LRHh/A8tpW7ru89lrlkU4AszXt1dbwSjVWguGrmlxE7tawVmDBlI1PILMkXAxJTwqhgsEeTHzj36D5CmHgQmNg==",
1009 | "requires": {
1010 | "atomic-sleep": "^1.0.0",
1011 | "flatstr": "^1.0.12"
1012 | }
1013 | },
1014 | "statuses": {
1015 | "version": "1.5.0",
1016 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz",
1017 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow="
1018 | },
1019 | "toidentifier": {
1020 | "version": "1.0.0",
1021 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz",
1022 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw=="
1023 | },
1024 | "type-is": {
1025 | "version": "1.6.18",
1026 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
1027 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
1028 | "requires": {
1029 | "media-typer": "0.3.0",
1030 | "mime-types": "~2.1.24"
1031 | }
1032 | },
1033 | "unpipe": {
1034 | "version": "1.0.0",
1035 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1036 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw="
1037 | },
1038 | "utils-merge": {
1039 | "version": "1.0.1",
1040 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
1041 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM="
1042 | },
1043 | "vary": {
1044 | "version": "1.1.2",
1045 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1046 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw="
1047 | },
1048 | "web-streams-polyfill": {
1049 | "version": "3.1.1",
1050 | "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.1.1.tgz",
1051 | "integrity": "sha512-Czi3fG883e96T4DLEPRvufrF2ydhOOW1+1a6c3gNjH2aIh50DNFBdfwh2AKoOf1rXvpvavAoA11Qdq9+BKjE0Q=="
1052 | }
1053 | }
1054 | }
1055 |
--------------------------------------------------------------------------------