Compare commits

...

7 Commits

71 changed files with 10055 additions and 1 deletions

BIN
.DS_Store vendored

Binary file not shown.

6
.gitignore vendored
View File

@ -2,4 +2,8 @@ CuestionarioKubernetes.docx
Videos de Clase Videos de Clase
Hojas de Asistencia Hojas de Asistencia
Clase 3/Clase 3.pptx Clase 3/Clase 3.pptx
Clase 3/Ejercicio/nginx-app Clase 3/Ejercicio/nginx-app
Clase 4/Clase 4.pptx
Clase 5/Ejercicio/server-graphql/node_modules
Clase 5/Ejercicio/server-graphql/build
Clase 5/Ejercicio/server-graphql/.env

BIN
Clase 1/.DS_Store vendored

Binary file not shown.

Binary file not shown.

BIN
Clase 2/.DS_Store vendored

Binary file not shown.

Binary file not shown.

BIN
Clase 4/.DS_Store vendored Normal file

Binary file not shown.

View File

@ -0,0 +1,42 @@
apiVersion: v1
kind: ConfigMap
metadata:
name: node-app-configmap
namespace: node-app
data:
# Mensaje personalizado
message: "Este es un mensaje personalizado."
# file-like keys
# Archivo de app (Código)
app.js: |
const express = require('express')
const app = express()
const port = process.env.PORT ? process.env.PORT : 80
app.get('/', (req, res) => {
res.send(process.env.MESSAGE ? process.env.MESSAGE : 'Hello World!')
})
app.get('/secret', (req, res) => {
res.send(process.env.SECRET ? process.env.SECRET : 'Hello Secret World!')
})
app.listen(port, () => {
console.log(`Example app listening on port ${port}`)
})
package.json: |
{
"name": "show-text-app",
"version": "1.0.0",
"description": "example app",
"main": "app.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "Alejandro Lembke Barrientos",
"license": "MIT",
"dependencies": {
"express": "^4.18.1"
}
}

View File

@ -0,0 +1,69 @@
apiVersion: apps/v1
kind: Deployment
metadata:
name: node-app
namespace: node-app
spec:
replicas: 1
selector:
matchLabels:
app: node-app
template:
metadata:
labels:
app: node-app
spec:
containers:
- name: node-app
image: ubuntu
command:
- "/bin/sh"
- "-c"
- "tail -f /dev/null"
env:
- name: PORT
value: "3000"
- name: SECRET
valueFrom:
secretKeyRef:
name: node-app-secret
key: secret
- name: MESSAGE
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: message
resources:
limits:
memory: "128Mi"
cpu: "500m"
ports:
- containerPort: 3000
lifecycle:
postStart:
exec:
command:
- "/bin/sh"
- "-c"
- "apt-get update"
- "&&"
- "apt-get install curl -y"
- "&&"
- "curl -sL https://deb.nodesource.com/setup_16.x -o nodesource_setup.sh"
- "&&"
- "bash nodesource_setup.sh"
- "&&"
- "apt install nodejs"
- "&&"
- "cd /app && npm install && node app.js"
volumeMounts:
- name: app-files
mountPath: /app/app.js
subPath: app.js
- name: app-files
mountPath: /app/package.json
subPath: package.json
volumes:
- name: app-files
configMap:
name: node-app-configmap

View File

@ -0,0 +1,4 @@
apiVersion: v1
kind: Namespace
metadata:
name: node-app

View File

@ -0,0 +1,8 @@
apiVersion: v1
kind: Secret
metadata:
name: node-app-secret
namespace: node-app
type: Opaque
data:
secret: RXN0ZSBlcyB1biB0ZXh0byBzZWNyZXRvLg==

View File

@ -0,0 +1,14 @@
apiVersion: v1
kind: Service
metadata:
name: node-app-svc
namespace: node-app
labels:
app: node-app
spec:
type: ClusterIP
selector:
app: node-app
ports:
- port: 80
targetPort: 3000

View File

@ -0,0 +1,4 @@
apiVersion: v1
kind: Namespace
metadata:
name: node-app

View File

@ -0,0 +1,13 @@
apiVersion: v1
kind: ConfigMap
metadata:
name: node-app-configmap
namespace: node-app
data:
#WHITELIST URLS Default to http://localhost
WHITELIST_URLS: http://localhost,http://localhost:4000
PORT: "4000"
HOST_MONGO: mongodb-node-app-svc
PORT_MONGO: "27017"
DB_MONGO: app_db

View File

@ -0,0 +1,9 @@
apiVersion: v1
kind: Secret
metadata:
name: node-app-secret
namespace: node-app
type: Opaque
data:
USER_MONGO: cm9vdA==
PASSWORD_MONGO: MTIzNA==

View File

@ -0,0 +1,16 @@
apiVersion: v1
kind: Service
metadata:
name: app-service
namespace: node-app
spec:
ports:
- name: 80-tcp
port: 80
protocol: TCP
targetPort: 4000
selector:
com.docker.project: node-app
type: LoadBalancer
status:
loadBalancer: {}

View File

@ -0,0 +1,13 @@
kind: Service
apiVersion: v1
metadata:
name: mongodb-node-app-svc
namespace: node-app
spec:
selector:
app: mongodb-node-app
ports:
- protocol: TCP
name: mongodb-node-app
port: 27017
targetPort: 27017

View File

@ -0,0 +1,58 @@
apiVersion: apps/v1
kind: Deployment
metadata:
name: node-app
namespace: node-app
spec:
replicas: 1
selector:
matchLabels:
com.docker.project: node-app
template:
metadata:
labels:
com.docker.project: node-app
spec:
containers:
- name: node-app
image: aleleba/example-app:2.0.1
imagePullPolicy: Always
env:
- name: PORT
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: PORT
- name: WHITELIST_URLS
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: WHITELIST_URLS
- name: HOST_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: HOST_MONGO
- name: PORT_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: PORT_MONGO
- name: DB_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: DB_MONGO
- name: USER_MONGO
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: PASSWORD_MONGO
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
ports:
- containerPort: 80
protocol: TCP

View File

@ -0,0 +1,49 @@
apiVersion: apps/v1
kind: StatefulSet
metadata:
name: mongodb-node-app
namespace: node-app
spec:
replicas: 1
serviceName: mongodb-node-app-svc
selector:
matchLabels:
app: mongodb-node-app
template:
metadata:
labels:
app: mongodb-node-app
spec:
containers:
- name: mongodb-node-app
image: mongo:5.0.12
imagePullPolicy: Always
ports:
- containerPort: 27017
env:
- name: MONGO_DATA_DIR
value: /data/db
- name: MONGO_LOG_DIR
value: /dev/null
- name: MONGODB_USER
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: MONGO_INITDB_ROOT_USERNAME
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: MONGODB_PASS
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
- name: MONGO_INITDB_ROOT_PASSWORD
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
- name: MONGO_INITDB_DATABASE
value: admin

View File

@ -0,0 +1,6 @@
{
"presets": [
"@babel/preset-env",
"@babel/preset-typescript"
]
}

View File

@ -0,0 +1,3 @@
#Quitar Node Modules
/node_modules
.env

View File

@ -0,0 +1,16 @@
#ENVIRONMENT Defauld production
ENV=development
#WHITELIST URLS Default to http://localhost
WHITELIST_URLS=http://localhost:4000
#GRAPHIQL Default to "false"
GRAPHIQL=false
#PLAYGROUND GRAPHQL Default to "false"
PLAYGROUND_GRAPHQL=true
# PORT EXPOSE APP Default to 4000
PORT=4000
# MONGO CONFIGURATION
HOST_MONGO=localhost
PORT_MONGO=27017
DB_MONGO=app_db
USER_MONGO=root
PASSWORD_MONGO=1234

View File

@ -0,0 +1,16 @@
#ENVIRONMENT Defauld production
ENV=
#WHITELIST URLS Default to http://localhost
WHITELIST_URLS=
#GRAPHIQL Default to "false"
GRAPHIQL=
#PLAYGROUND GRAPHQL Default to "false"
PLAYGROUND_GRAPHQL=
# PORT EXPOSE APP Default to 4000
PORT=
# MONGO CONFIGURATION
HOST_MONGO=
PORT_MONGO=
DB_MONGO=
USER_MONGO=
PASSWORD_MONGO=

View File

@ -0,0 +1,7 @@
#Eslint
.eslintrc.js
#Build
build
#Webpack
webpack.config.ts
webpack.config.dev.ts

View File

@ -0,0 +1,37 @@
module.exports = {
'env': {
'browser': true,
'es2021': true,
'node': true,
},
'extends': [
'eslint:recommended',
'plugin:@typescript-eslint/recommended'
],
'parser': '@typescript-eslint/parser',
'parserOptions': {
'ecmaVersion': 'latest',
'sourceType': 'module'
},
'plugins': [
'@typescript-eslint'
],
'rules': {
'indent': [
'error',
'tab'
],
'linebreak-style': [
'error',
'unix'
],
'quotes': [
'error',
'single'
],
'semi': [
'error',
'always'
]
}
};

View File

@ -0,0 +1,4 @@
# dependencies
/node_modules
/build
.env

View File

@ -0,0 +1,3 @@
{
"typescript.tsdk": "node_modules/typescript/lib"
}

View File

@ -0,0 +1,54 @@
FROM aleleba/ubuntu:4.0.0 as dev-builder
#docker build --target dev-builder -t aleleba/app:dev .
#docker run --name mongodb --rm -p 27017:27017 -e MONGODB_USER=root -e MONGODB_PASS=1234 -e MONGO_INITDB_ROOT_USERNAME=root -e MONGO_INITDB_ROOT_PASSWORD=1234 -d mongo:5.0.12
RUN sudo apt-get update
COPY ["./package.json", "./package-lock.json", "/app/"]
WORKDIR /app
RUN npm install --legacy-peer-deps
RUN npm audit fix --legacy-peer-deps
COPY [".", "/app/"]
EXPOSE 80
CMD ["/bin/bash"]
#CURRENT_DIR=$(pwd) && docker run -it -d --rm --name app -p 80:80 -v "$CURRENT_DIR":/app aleleba/app:dev
FROM aleleba/ubuntu:4.0.0 as pre-prod-builder
RUN sudo apt-get update
COPY --from=dev-builder ["/app/", "/app/"]
RUN sudo rm -rf /app/.env
WORKDIR /app
RUN npm run build
FROM aleleba/ubuntu:4.0.0 as prod-builder
#docker build -t aleleba/ro-ut:tag .
RUN sudo apt-get update
WORKDIR /app
COPY --from=pre-prod-builder ["/app/package.json", "/app/package-lock.json", "/app/"]
COPY --from=pre-prod-builder ["/app/node_modules/", "/app/node_modules/"]
COPY --from=pre-prod-builder ["/app/build/", "/app/build/"]
EXPOSE 80
#docker run -it -d --rm -p 4000:4000 -e WHITELIST_URLS=http://localhost:4000 -e PLAYGROUND_GRAPHQL=true -e HOST_MONGO=localhost -e PORT_MONGO=27017 -e DB_MONGO=app_db -e USER_MONGO=root -e PASSWORD_MONGO=1234 --name app aleleba/example-app:2.0.1
CMD ["npm", "start"]

View File

@ -0,0 +1,21 @@
MIT License
Copyright (c) 2022 Alejandro Lembke Barrientos
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

View File

@ -0,0 +1,11 @@
const PRName = function () {
let ID = '';
// let characters = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
const characters = '0123456789';
for ( let i = 0; i < 6; i++ ) {
ID += characters.charAt(Math.floor(Math.random() * 10));
}
return 'PR-'+ID;
};
console.log(PRName());

View File

@ -0,0 +1,76 @@
# Create Node TS GraphQL Server
This project aims to have a starter kit for creating a new Node with typescript, GraphQL server and tools that generally go along with it.
Tech(Library or Framework) | Version |
--- | --- |
Jest (Testing) | 29.0.3
Typescript | 4.8.3
GraphQL | 16.6.0
## Setup
To create a new project run in the terminal:
```
npx @aleleba/create-node-ts-graphql-server server-app-name
```
Then run:
```
cd server-app-name
```
You will need to create a new .env file at the root of the project for global config.
This is an example of config.
```
#ENVIRONMENT Defauld production
ENVIRONMENT=development
#WHITELIST URLS Default to http://localhost
WHITELIST_URLS=https://someurl.com
#GRAPHIQL Default to "false"
GRAPHIQL=true
#PLAYGROUND GRAPHQL Default to "false"
PLAYGROUND_GRAPHQL=true
# PORT EXPOSE APP Default to 4000
PORT=4000
```
The default environment is production, the server-app port defauld is 4000, the default whitelist is http://localhost and the default graphiql is false.
### For Development
In the terminal run:
```
npm run start:dev
```
The ENV enviroment variable should be "development" and choose the port of your preference with the enviroment variable PORT.
You will find the controllers on:
```
scr/controllers/
```
You will find the models on:
```
scr/models
```
You will find the GraphQL server, resolvers and schema definition on:
```
scr/GraphQL
```
The manage of the routes for custom API you should find on:
```
scr/routes
```
This will start the app in development mode, also use nodemon and webpack to real time coding!
Enjoy coding!
### For Production
In the terminal run:
```
npm run build
```
It will create a build folder and run:
```
npm start
```
This will start the app.
## Cheers
Hope you enjoy this proyect! Sincerely Alejandro Lembke Barrientos.

View File

@ -0,0 +1,26 @@
import * as dotenv from 'dotenv';
dotenv.config();
export const deFaultValues = {
ENV: 'production',
GRAPHIQL: 'false',
PLAYGROUND_GRAPHQL: 'true',
WHITELIST_URLS: 'http://localhost,http://localhost:4000',
PORT: 4000,
};
export const config = {
ENV: process.env.ENV,
GRAPHIQL: process.env.GRAPHIQL === 'true' ? true : false,
PLAYGROUND_GRAPHQL: process.env.PLAYGROUND_GRAPHQL === 'true' ? true : false,
WHITELIST_URLS: process.env.WHITELIST_URLS ? process.env.WHITELIST_URLS.split(',') : deFaultValues.WHITELIST_URLS,
PORT: process.env.PORT,
mongoDBConfig: {
host : process.env.HOST_MONGO,
port : process.env.PORT_MONGO,
db : process.env.DB_MONGO,
user: encodeURIComponent(process.env.USER_MONGO),
password: encodeURIComponent(process.env.PASSWORD_MONGO)
},
};

View File

@ -0,0 +1,6 @@
module.exports = {
testEnvironment: 'node',
transform: {
"^.+\\.ts$": "ts-jest"
},
};

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,78 @@
{
"name": "server-graphql",
"version": "1.0.0",
"description": "Node with Typescript and GraphQL Server",
"main": "index.js",
"scripts": {
"start": "node build/index.js",
"start:dev": "webpack-cli --config webpack.config.dev.ts",
"start:nodemon": "nodemon build/index.js",
"build": "webpack-cli --config webpack.config.ts",
"lint": "eslint ./ --ext .js --ext .ts",
"lint:fix": "eslint ./ --ext .js --ext .ts --fix",
"test": "jest",
"test:watch": "jest --watch"
},
"repository": {
"type": "git",
"url": "git+https://github.com/aleleba/node-ts-graphql-server.git"
},
"keywords": [
"node",
"express",
"typescript",
"graphql",
"server"
],
"author": "Alejandro Lembke Barrientos",
"license": "MIT",
"bugs": {
"url": "https://github.com/aleleba/node-ts-graphql-server/issues"
},
"homepage": "https://github.com/aleleba/node-ts-graphql-server#readme",
"dependencies": {
"@graphql-tools/schema": "^9.0.4",
"body-parser": "^1.20.0",
"cookie-parser": "^1.4.6",
"cors": "^2.8.5",
"dotenv": "^16.0.2",
"express": "^4.18.1",
"express-graphql": "^0.12.0",
"graphql": "^16.6.0",
"graphql-playground-middleware-express": "^1.7.23",
"graphql-subscriptions": "^2.0.0",
"graphql-tools": "^8.3.6",
"graphql-ws": "^5.10.2",
"mongodb": "^4.9.1",
"web-push": "^3.5.0",
"ws": "^8.8.1"
},
"devDependencies": {
"@babel/core": "^7.19.1",
"@babel/preset-env": "^7.19.1",
"@babel/preset-typescript": "^7.18.6",
"@babel/register": "^7.18.9",
"@types/jest": "^29.0.2",
"@types/node": "^18.7.18",
"@types/webpack": "^5.28.0",
"@types/webpack-node-externals": "^2.5.3",
"@typescript-eslint/eslint-plugin": "^5.37.0",
"@typescript-eslint/parser": "^5.37.0",
"babel-loader": "^8.2.5",
"clean-webpack-plugin": "^4.0.0",
"compression-webpack-plugin": "^10.0.0",
"eslint": "^8.23.1",
"eslint-webpack-plugin": "^3.2.0",
"jest": "^29.0.3",
"nodemon": "^2.0.19",
"supertest": "^6.2.4",
"ts-jest": "^29.0.1",
"ts-loader": "^9.3.1",
"typescript": "^4.8.3",
"webpack": "^5.74.0",
"webpack-cli": "^4.10.0",
"webpack-manifest-plugin": "^5.0.0",
"webpack-node-externals": "^3.0.0",
"webpack-shell-plugin-next": "^2.2.2"
}
}

View File

@ -0,0 +1,5 @@
// index.d.ts
declare module "*.gql" {
const content: any;
export default content;
}

View File

@ -0,0 +1,21 @@
'use strict';
import { createProductoController, deleteProductoController, getProductosController } from '../../controllers/controllerGraphQL';
// A map of functions which return data for the schema.
const resolvers = {
Query: {
// eslint-disable-next-line
productos: (rootValue, args, context) => getProductosController(),
},
Mutation: {
// eslint-disable-next-line
productosMutation: (rootValue, args, context) => ({}),
},
ProductosMutation: {
insertProducto: (rootValue, args) => createProductoController(rootValue, args),
deleteProducto: (rootValue, args) => deleteProductoController(rootValue, args)
}
};
export default resolvers;

View File

@ -0,0 +1,24 @@
module.exports = `
"""Producto Query"""
type Producto {
id: ID
nombre: String
}
type InsertProductoRes {
producto: Producto
mensaje: String
}
input ProductoInput{
nombre: String
}
"""Esta es la Data de LogIn, Si los datos no son correctos devuelve el usuario Null y la conexion en False"""
type ProductosMutation {
insertProducto(producto: ProductoInput): InsertProductoRes
deleteProducto(id: ID): String
}
`

View File

@ -0,0 +1,17 @@
import { makeExecutableSchema } from '@graphql-tools/schema';
import resolvers from'../resolvers';
import Productos from './Productos.gql';
// The GraphQL schema
const rootTypes = `
type Query {
productos: [Producto]
}
type Mutation {
productosMutation: ProductosMutation
}
`;
const typeDefs = [ rootTypes, Productos ];
export default makeExecutableSchema({typeDefs, resolvers});

View File

@ -0,0 +1,30 @@
'use strict';
import express from 'express'; //express
import { graphqlHTTP } from 'express-graphql';
import { config } from '../../config';
import schema from './schema';
const server = express.Router();//Router de Express
server.use(
'/',
graphqlHTTP( (req, res) => {
return {
schema,
graphiql: config.GRAPHIQL,
context: { req, res }
};
}),
);
// DO NOT DO app.listen() unless we're testing this directly
if (require.main === module) {
server.listen((process.env.PORT || 4000), () => {
console.log(`Iniciando Express en el puerto 4000${server.graphqlPath}`); /*${app.get('port')}*/
});
}
// Instead do export the app:
export default server;

View File

@ -0,0 +1,124 @@
'use strict';
import { ObjectID } from 'mongodb';
import ProductosModel from '../../models';
const { insertProducto, getAllProductos, deleteProducto } = ProductosModel;
//Funcion para insertar un Producto en MongoDB
export const createProductoController = async (rootValue, args) => {
const producto = args.producto;
const promise = new Promise((resolve) => {
const cb = (res, disconnect) => {
//assert.equal(err, null);
//console.log("Found the following records");
console.log(res.insertedId.toString());
resolve({
producto: {
id: res.insertedId.toString(),
...producto
},
mensaje: 'Se inserto exitosamente el Producto'
});
//console.log(res)
disconnect();
};
insertProducto(producto, cb);
});
const result = await promise; // wait till the promise resolves (*)
//console.log(result); // "done!"
return result;
};
//Termina Funcion para insertar un Producto en MongoDB
//Funcion para traer todos los Productos
export const getProductosController = async () => {
const promise = new Promise((resolve) => {
const cb = (err, res, disconnect) => {
if(err){
new Error('No hay registros de Productos');
disconnect();
}else{
const data = res.map( producto => {
const productoReturn = {
...producto,
id: producto._id,
};
delete productoReturn['_id'];
return productoReturn;
});
resolve(data);
disconnect();
}
};
getAllProductos(cb);
});
const result = await promise; // wait till the promise resolves (*)
//console.log(result); // "done!"
return result;
};
//Termina Funcion para traer todos los Productos
//Funcion para Eliminar un Objetivo en MongoDB
export const deleteProductoController = async (rootValue, args) => {
const id = args.id;
const promise = new Promise((resolve) => {
const cb = (res, disconnect) => {
//console.log(res)
if (res) {
//assert.equal(1, res.deletedCount);
resolve(`Se eliminó exitosamente el Producto con id: ${id}`);
disconnect();
} else {
resolve('No se eliminó el Producto');
disconnect();
}
};
deleteProducto(cb, new ObjectID(id));
});
const result = await promise; // wait till the promise resolves (*)
//console.log(result); // "done!"
return result;
};
//Termina Funcion para Eliminar un Objetivo en MongoDB

View File

@ -0,0 +1,84 @@
'use strict';
import ws from 'ws'; // yarn add ws
import express from 'express'; //express
import cors from 'cors';
import cookieParser from 'cookie-parser';
import { useServer } from 'graphql-ws/lib/use/ws';
import { execute, subscribe } from 'graphql';
import GraphQLserver from './GraphQL/server';// Server of GraphQL,
import expressPlayground from 'graphql-playground-middleware-express';
import schema from './GraphQL/schema';
import { config } from '../config';
import apiRouter from './routes';
const app = express(), //creating app
whitelist = config.WHITELIST_URLS,
corsOptions = {
origin: function (origin, callback) {
if (whitelist.indexOf(origin) !== -1 || !origin) {
callback(null, true);
} else {
callback(new Error('Not allowed by CORS'));
}
},
credentials: true
};
//Inicialization of services of express
app
.use(cookieParser())
.use(express.urlencoded({limit: '500mb', extended: true}))
.use(express.json({limit: '500mb', extended: true}))
.use(cors(corsOptions))
.use(apiRouter)//Routes de App
.use('/graphql', GraphQLserver);//Server of Graphql
if(config.PLAYGROUND_GRAPHQL === true){
app.get('/playground', expressPlayground({ endpoint: '/graphql' }));
}
// DO NOT DO app.listen() unless we're testing this directly
if (require.main === module) {
const server = app.listen(config.PORT, () => {
// create and use the websocket server
const wsServer = new ws.Server({
server,
path: '/graphql',
});
useServer({
schema,
execute,
subscribe,
// eslint-disable-next-line
onConnect: (ctx) => {
//console.log('Connect');
},
// eslint-disable-next-line
onSubscribe: (ctx, msg) => {
//console.log('Subscribe');
},
// eslint-disable-next-line
onNext: (ctx, msg, args, result) => {
//console.debug('Next');
},
// eslint-disable-next-line
onError: (ctx, msg, errors) => {
//console.error('Error');
},
// eslint-disable-next-line
onComplete: (ctx, msg) => {
//console.log('Complete');
},
}, wsServer);
console.log(`Starting Express on port ${config.PORT} and iniciating server of web sockets`);
});
}
// Instead do export the app:
export default app;

View File

@ -0,0 +1,38 @@
/* eslint-disable */
'use strict';
import { MongoClient } from 'mongodb';
import { config } from '../../../../config';
// Connection URL
// eslint-disable-next-line no-useless-escape
const url = `mongodb:\/\/${config.mongoDBConfig.user}:${config.mongoDBConfig.password}@${config.mongoDBConfig.host}:${parseInt(config.mongoDBConfig.port)}`;
const dbName = config.mongoDBConfig.db;
async function connect(cb) {
const client = new MongoClient(url, { useNewUrlParser: true, useUnifiedTopology: true })
try{
await client.connect();
console.log('conexion exitosa con mongoDB');
const db = client.db(dbName);
const disconnect = () => {
// Close connection
client.close( ()=>{
console.log('desconexion exitosa con mongoDB');
});
};
await cb(db, disconnect);
}catch(err){
console.log(err.stack);
}
}
export default connect;
/* eslint-enable */

View File

@ -0,0 +1,51 @@
'use strict';
import conn from './config/apiMongoDB-connection';
export const insertProducto = (data, cb) => {
async function callback (db, disconnect) {
// Insert a single document
await db.collection('Productos')
.insertOne(data)
.then( res => cb(res, disconnect));
}
conn(callback);
};
export const getAllProductos = (cb) => {
async function callback(db, disconnect) {
// Insert a single document
await db.collection('Productos')
.find()
.limit(0)
.toArray( (err, res) => {
cb(err, res, disconnect);
});
}
conn(callback);
};
export const deleteProducto = (cb, id) => {
async function callback (db, disconnect) {
// Insert a single document
await db.collection('Productos')
.deleteOne({ _id: id })
.then( res => cb(res, disconnect));
//.updateOne({ _id: data.id }, { $set: { estado: data.estado} })
}
conn(callback);
};

View File

@ -0,0 +1,5 @@
'use strict';
import { insertProducto, getAllProductos, deleteProducto } from './apiMongoDBModel';
export default { insertProducto, getAllProductos, deleteProducto };

View File

@ -0,0 +1,13 @@
'use strict';
// use this to set API REST
import express from 'express';
import bodyParser from 'body-parser';//bodyParser conversionde Api REST,
const apiRouter = express.Router();//Router de Express
apiRouter
.use(bodyParser.json())
.use(bodyParser.urlencoded({extended: false}));
export default apiRouter;

View File

@ -0,0 +1,11 @@
{
"compilerOptions": {
"module": "commonjs",
"esModuleInterop": true,
"target": "es6",
"moduleResolution": "node",
"sourceMap": true,
"typeRoots" : ["./src/@types", "./node_modules/@types"],
},
"lib": ["es2015"]
}

View File

@ -0,0 +1,64 @@
import path from 'path';
import webpack from 'webpack';
import { CleanWebpackPlugin } from 'clean-webpack-plugin';
import ESLintPlugin from 'eslint-webpack-plugin';
import nodeExternals from 'webpack-node-externals';
import WebpackShellPluginNext from 'webpack-shell-plugin-next';
import { deFaultValues } from './config';
const ROOT_DIR = path.resolve(__dirname);
const resolvePath = (...args) => path.resolve(ROOT_DIR, ...args);
const BUILD_DIR = resolvePath('build');
const config = {
entry: './src/index.ts',
target: 'node',
watch: true,
externals: [nodeExternals()],
output: {
path: BUILD_DIR,
filename: 'index.js',
},
resolve: {
extensions: ['.js', '.ts', '.json', '.gql'],
alias: {
'@controllers': path.resolve(__dirname, 'controllers/'),
'@models': path.resolve(__dirname, 'models/'),
'@controllerGraphQL': path.resolve(__dirname, 'controllers/controllerGraphQL/'),
'@GraphQL': path.resolve(__dirname, 'GraphQL/'),
'@config': path.resolve(__dirname, 'config/'),
}
},
mode: 'development',
module: {
rules: [
{
test: /\.(js|ts|mjs|gql)$/,
exclude: /node_modules/,
use: {
loader: 'babel-loader',
},
},
{
test: /\.(ts)$/, loader: 'ts-loader',
exclude: /node_modules/
},
],
},
plugins: [
new CleanWebpackPlugin(),
new ESLintPlugin(),
new webpack.EnvironmentPlugin({
...deFaultValues
}),
new WebpackShellPluginNext({
onBuildEnd: {
scripts: ['nodemon build/index.js'],
blocking: false,
parallel: true
}
})
],
};
export default config;

View File

@ -0,0 +1,62 @@
import path from 'path';
import webpack from 'webpack';
import TerserPlugin from 'terser-webpack-plugin';
import { CleanWebpackPlugin } from 'clean-webpack-plugin';
import ESLintPlugin from 'eslint-webpack-plugin';
import nodeExternals from 'webpack-node-externals';
import { deFaultValues } from './config';
const ROOT_DIR = path.resolve(__dirname);
const resolvePath = (...args) => path.resolve(ROOT_DIR, ...args);
const BUILD_DIR = resolvePath('build');
const config = {
entry: './src/index.ts',
target: 'node',
externals: [nodeExternals()],
output: {
path: BUILD_DIR,
filename: 'index.js',
},
resolve: {
extensions: ['.js', '.ts', '.json', '.gql'],
alias: {
'@controllers': path.resolve(__dirname, 'controllers/'),
'@models': path.resolve(__dirname, 'models/'),
'@controllerGraphQL': path.resolve(__dirname, 'controllers/controllerGraphQL/'),
'@GraphQL': path.resolve(__dirname, 'GraphQL/'),
'@config': path.resolve(__dirname, 'config/'),
}
},
mode: 'production',
module: {
rules: [
{
test: /\.(js|ts|mjs|gql)$/,
exclude: /node_modules/,
use: {
loader: 'babel-loader',
},
},
{
test: /\.(ts)$/, loader: "ts-loader",
exclude: /node_modules/
},
],
},
plugins: [
new CleanWebpackPlugin(),
new ESLintPlugin(),
new webpack.EnvironmentPlugin({
...deFaultValues
}),
],
optimization: {
minimize: true,
minimizer: [
new TerserPlugin(),
],
},
};
export default config;

BIN
Clase 6/.DS_Store vendored Normal file

Binary file not shown.

BIN
Clase 6/Clase 6.pptx Normal file

Binary file not shown.

View File

@ -0,0 +1,4 @@
apiVersion: v1
kind: Namespace
metadata:
name: node-app

View File

@ -0,0 +1,6 @@
kind: StorageClass
apiVersion: storage.k8s.io/v1
metadata:
name: my-local-storage
provisioner: kubernetes.io/no-provisioner
volumeBindingMode: WaitForFirstConsumer

View File

@ -0,0 +1,21 @@
apiVersion: v1
kind: PersistentVolume
metadata:
name: my-local-pv
spec:
capacity:
storage: 3Gi
accessModes:
- ReadWriteOnce
persistentVolumeReclaimPolicy: Retain
storageClassName: my-local-storage
local:
path: /mnt
nodeAffinity:
required:
nodeSelectorTerms:
- matchExpressions:
- key: kubernetes.io/hostname
operator: In
values:
- docker-desktop

View File

@ -0,0 +1,12 @@
kind: PersistentVolumeClaim
apiVersion: v1
metadata:
name: node-app-mongodb-pvc
namespace: node-app
spec:
storageClassName: my-local-storage
accessModes:
- ReadWriteOnce
resources:
requests:
storage: 3Gi

View File

@ -0,0 +1,13 @@
apiVersion: v1
kind: ConfigMap
metadata:
name: node-app-configmap
namespace: node-app
data:
#WHITELIST URLS Default to http://localhost
WHITELIST_URLS: http://localhost,http://localhost:4000
PORT: "4000"
HOST_MONGO: mongodb-node-app-svc
PORT_MONGO: "27017"
DB_MONGO: app_db

View File

@ -0,0 +1,9 @@
apiVersion: v1
kind: Secret
metadata:
name: node-app-secret
namespace: node-app
type: Opaque
data:
USER_MONGO: cm9vdA==
PASSWORD_MONGO: MTIzNA==

View File

@ -0,0 +1,16 @@
apiVersion: v1
kind: Service
metadata:
name: app-service
namespace: node-app
spec:
ports:
- name: 80-tcp
port: 80
protocol: TCP
targetPort: 4000
selector:
com.docker.project: node-app
type: LoadBalancer
status:
loadBalancer: {}

View File

@ -0,0 +1,13 @@
kind: Service
apiVersion: v1
metadata:
name: mongodb-node-app-svc
namespace: node-app
spec:
selector:
app: mongodb-node-app
ports:
- protocol: TCP
name: mongodb-node-app
port: 27017
targetPort: 27017

View File

@ -0,0 +1,58 @@
apiVersion: apps/v1
kind: Deployment
metadata:
name: node-app
namespace: node-app
spec:
replicas: 1
selector:
matchLabels:
com.docker.project: node-app
template:
metadata:
labels:
com.docker.project: node-app
spec:
containers:
- name: node-app
image: aleleba/example-app:2.0.1
imagePullPolicy: Always
env:
- name: PORT
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: PORT
- name: WHITELIST_URLS
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: WHITELIST_URLS
- name: HOST_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: HOST_MONGO
- name: PORT_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: PORT_MONGO
- name: DB_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: DB_MONGO
- name: USER_MONGO
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: PASSWORD_MONGO
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
ports:
- containerPort: 80
protocol: TCP

View File

@ -0,0 +1,56 @@
apiVersion: apps/v1
kind: StatefulSet
metadata:
name: mongodb-node-app
namespace: node-app
spec:
replicas: 1
serviceName: mongodb-node-app-svc
selector:
matchLabels:
app: mongodb-node-app
template:
metadata:
labels:
app: mongodb-node-app
spec:
containers:
- name: mongodb-node-app
image: mongo:5.0.12
imagePullPolicy: Always
ports:
- containerPort: 27017
env:
- name: MONGO_DATA_DIR
value: /data/db
- name: MONGO_LOG_DIR
value: /dev/null
- name: MONGODB_USER
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: MONGO_INITDB_ROOT_USERNAME
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: MONGODB_PASS
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
- name: MONGO_INITDB_ROOT_PASSWORD
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
- name: MONGO_INITDB_DATABASE
value: admin
volumeMounts:
- mountPath: /data/db
name: node-app-mongodb-pvc
volumes:
- name: node-app-mongodb-pvc
persistentVolumeClaim:
claimName: node-app-mongodb-pvc

View File

@ -0,0 +1 @@
kubectl delete all --all -n {namespace}

View File

@ -0,0 +1,11 @@
install helm first:
docs: https://helm.sh/docs/intro/install/
#Before you can install the chart you will need to add the metrics-server repo to Helm.
1. helm repo add metrics-server https://kubernetes-sigs.github.io/metrics-server/
#After you've installed the repo you can install the chart.
2. helm upgrade --install metrics-server metrics-server/metrics-server --namespace kube-system
On docker desktop add: --set args={"--kubelet-insecure-tls=true"} --set containerPort=443
Link: https://artifacthub.io/packages/helm/metrics-server/metrics-server

View File

@ -0,0 +1,198 @@
apiVersion: v1
kind: ServiceAccount
metadata:
labels:
k8s-app: metrics-server
name: metrics-server
namespace: kube-system
---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
labels:
k8s-app: metrics-server
rbac.authorization.k8s.io/aggregate-to-admin: "true"
rbac.authorization.k8s.io/aggregate-to-edit: "true"
rbac.authorization.k8s.io/aggregate-to-view: "true"
name: system:aggregated-metrics-reader
rules:
- apiGroups:
- metrics.k8s.io
resources:
- pods
- nodes
verbs:
- get
- list
- watch
---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
labels:
k8s-app: metrics-server
name: system:metrics-server
rules:
- apiGroups:
- ""
resources:
- nodes/metrics
verbs:
- get
- apiGroups:
- ""
resources:
- pods
- nodes
verbs:
- get
- list
- watch
---
apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
labels:
k8s-app: metrics-server
name: metrics-server-auth-reader
namespace: kube-system
roleRef:
apiGroup: rbac.authorization.k8s.io
kind: Role
name: extension-apiserver-authentication-reader
subjects:
- kind: ServiceAccount
name: metrics-server
namespace: kube-system
---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRoleBinding
metadata:
labels:
k8s-app: metrics-server
name: metrics-server:system:auth-delegator
roleRef:
apiGroup: rbac.authorization.k8s.io
kind: ClusterRole
name: system:auth-delegator
subjects:
- kind: ServiceAccount
name: metrics-server
namespace: kube-system
---
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRoleBinding
metadata:
labels:
k8s-app: metrics-server
name: system:metrics-server
roleRef:
apiGroup: rbac.authorization.k8s.io
kind: ClusterRole
name: system:metrics-server
subjects:
- kind: ServiceAccount
name: metrics-server
namespace: kube-system
---
apiVersion: v1
kind: Service
metadata:
labels:
k8s-app: metrics-server
name: metrics-server
namespace: kube-system
spec:
ports:
- name: https
port: 443
protocol: TCP
targetPort: https
selector:
k8s-app: metrics-server
---
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
k8s-app: metrics-server
name: metrics-server
namespace: kube-system
spec:
selector:
matchLabels:
k8s-app: metrics-server
strategy:
rollingUpdate:
maxUnavailable: 0
template:
metadata:
labels:
k8s-app: metrics-server
spec:
containers:
- args:
- --cert-dir=/tmp
- --secure-port=4443
- --kubelet-preferred-address-types=InternalIP,ExternalIP,Hostname
- --kubelet-use-node-status-port
- --metric-resolution=15s
- --authorization-always-allow-paths=/livez,/readyz
- --kubelet-insecure-tls
image: k8s.gcr.io/metrics-server/metrics-server:v0.6.1
imagePullPolicy: IfNotPresent
livenessProbe:
failureThreshold: 3
httpGet:
path: /livez
port: https
scheme: HTTPS
periodSeconds: 10
name: metrics-server
ports:
- containerPort: 4443
name: https
protocol: TCP
readinessProbe:
failureThreshold: 3
httpGet:
path: /readyz
port: https
scheme: HTTPS
initialDelaySeconds: 20
periodSeconds: 10
resources:
requests:
cpu: 100m
memory: 200Mi
securityContext:
allowPrivilegeEscalation: false
readOnlyRootFilesystem: true
runAsNonRoot: true
runAsUser: 1000
volumeMounts:
- mountPath: /tmp
name: tmp-dir
nodeSelector:
kubernetes.io/os: linux
priorityClassName: system-cluster-critical
serviceAccountName: metrics-server
volumes:
- emptyDir: {}
name: tmp-dir
---
apiVersion: apiregistration.k8s.io/v1
kind: APIService
metadata:
labels:
k8s-app: metrics-server
name: v1beta1.metrics.k8s.io
spec:
group: metrics.k8s.io
groupPriorityMinimum: 100
insecureSkipTLSVerify: true
service:
name: metrics-server
namespace: kube-system
version: v1beta1
versionPriority: 100

View File

@ -0,0 +1,13 @@
apiVersion: v1
kind: ConfigMap
metadata:
name: node-app-configmap
namespace: default
data:
#WHITELIST URLS Default to http://localhost
WHITELIST_URLS: http://localhost,http://localhost:4000
PORT: "4000"
HOST_MONGO: mongodb-node-app-svc
PORT_MONGO: "27017"
DB_MONGO: app_db

View File

@ -0,0 +1,9 @@
apiVersion: v1
kind: Secret
metadata:
name: node-app-secret
namespace: default
type: Opaque
data:
USER_MONGO: cm9vdA==
PASSWORD_MONGO: MTIzNA==

View File

@ -0,0 +1,16 @@
apiVersion: v1
kind: Service
metadata:
name: app-service
namespace: default
spec:
ports:
- name: 80-tcp
port: 80
protocol: TCP
targetPort: 4000
selector:
com.docker.project: node-app
type: LoadBalancer
status:
loadBalancer: {}

View File

@ -0,0 +1,13 @@
kind: Service
apiVersion: v1
metadata:
name: mongodb-node-app-svc
namespace: default
spec:
selector:
app: mongodb-node-app
ports:
- protocol: TCP
name: mongodb-node-app
port: 27017
targetPort: 27017

View File

@ -0,0 +1,65 @@
apiVersion: apps/v1
kind: Deployment
metadata:
name: node-app
namespace: default
spec:
replicas: 1
selector:
matchLabels:
com.docker.project: node-app
template:
metadata:
labels:
com.docker.project: node-app
spec:
containers:
- name: node-app
image: aleleba/example-app:2.0.1
imagePullPolicy: Always
resources:
requests:
memory: "256Mi"
cpu: "800m"
limits:
memory: "512Mi"
cpu: "2000m"
env:
- name: PORT
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: PORT
- name: WHITELIST_URLS
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: WHITELIST_URLS
- name: HOST_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: HOST_MONGO
- name: PORT_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: PORT_MONGO
- name: DB_MONGO
valueFrom:
configMapKeyRef:
name: node-app-configmap
key: DB_MONGO
- name: USER_MONGO
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: PASSWORD_MONGO
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
ports:
- containerPort: 80
protocol: TCP

View File

@ -0,0 +1,49 @@
apiVersion: apps/v1
kind: StatefulSet
metadata:
name: mongodb-node-app
namespace: default
spec:
replicas: 1
serviceName: mongodb-node-app-svc
selector:
matchLabels:
app: mongodb-node-app
template:
metadata:
labels:
app: mongodb-node-app
spec:
containers:
- name: mongodb-node-app
image: mongo:5.0.12
imagePullPolicy: Always
ports:
- containerPort: 27017
env:
- name: MONGO_DATA_DIR
value: /data/db
- name: MONGO_LOG_DIR
value: /dev/null
- name: MONGODB_USER
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: MONGO_INITDB_ROOT_USERNAME
valueFrom:
secretKeyRef:
name: node-app-secret
key: USER_MONGO
- name: MONGODB_PASS
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
- name: MONGO_INITDB_ROOT_PASSWORD
valueFrom:
secretKeyRef:
name: node-app-secret
key: PASSWORD_MONGO
- name: MONGO_INITDB_DATABASE
value: admin

View File

@ -0,0 +1,19 @@
apiVersion: autoscaling/v2
kind: HorizontalPodAutoscaler
metadata:
name: node-app
namespace: default
spec:
scaleTargetRef:
apiVersion: apps/v1
kind: Deployment
name: node-app
minReplicas: 1
maxReplicas: 5
metrics:
- type: Resource
resource:
name: cpu
target:
type: Utilization
averageUtilization: 30