Compare commits

..

44 Commits

Author SHA1 Message Date
Leifer Mendez
327cf5730b Merge pull request #137 from vicente1992/feat-mysql-adapter
feat(adapter): added adapter mysql
2022-12-03 12:06:03 +01:00
vicente1992
717a7dc95f feat(adapter): added adapter mysql 2022-12-02 09:41:57 -05:00
Leifer Mendez
7d41699207 docs(contributing): update 2022-12-01 21:33:50 +01:00
Leifer Mendez
468a2ba251 chore(cli): 🔥 fix 2022-12-01 21:07:00 +01:00
Leifer Mendez
50d73f7bc8 chore(cli): 🎨 remove uneccesary steps 2022-12-01 20:58:58 +01:00
Leifer Mendez
05c6fd4528 chore(cli): added new function 2022-12-01 20:44:12 +01:00
Leifer Mendez
648354500b fix(fix): fix 2022-12-01 18:42:18 +01:00
Leifer Mendez
28c0480b8b fix(fix): fix 2022-12-01 18:39:49 +01:00
Leifer Mendez
903b4d79ac docs(contributing): 📝 added more example 2022-12-01 18:17:19 +01:00
Leifer Mendez
026c189901 Merge pull request #130 from leifermendez/dev
Dev
2022-12-01 12:52:46 +01:00
Leifer
f2b30ee349 docs: 📝 update CONTRIBUTING 2022-12-01 12:46:49 +01:00
Leifer
5c02a9325a chore(release): 0.2.0-alpha.0 2022-12-01 12:38:17 +01:00
Leifer
7645c8642f test 2022-12-01 12:33:18 +01:00
Leifer
c5ebbe319f refactor(hook): added new improvement 2022-12-01 12:32:45 +01:00
Leifer
6f36eb1690 test 2022-12-01 10:35:41 +01:00
Leifer
18f9e006a3 test 2022-12-01 10:34:30 +01:00
Leifer
a7e334ebe9 add banner 2022-12-01 10:27:18 +01:00
Leifer
a5e15d9d84 edit hook 2022-12-01 10:17:19 +01:00
Leifer
b3173517b4 add pretty-quick 2022-12-01 10:16:06 +01:00
Leifer Mendez
06d2963163 Merge pull request #129 from leifermendez/feature/monorepo
Feature/monorepo
2022-12-01 09:53:28 +01:00
Leifer Mendez
df8282015d docs(example-app): add cli crate app 2022-11-30 21:39:30 +01:00
Leifer Mendez
81b0aab850 refactor(io): added new method addChild 2022-11-30 21:06:26 +01:00
Leifer Mendez
a8705c5b44 refactor(io): added new method addChild 2022-11-30 21:05:32 +01:00
Leifer Mendez
efe739f9fc refactor(io): added new method addChild 2022-11-30 21:04:45 +01:00
Leifer Mendez
2e83a0508a docs(contribuiting): update 2022-11-30 17:14:50 +01:00
Leifer Mendez
d66adb2a1f Merge pull request #126 from leifermendez/feature/monorepo
Feature/monorepo
2022-11-29 22:14:56 +01:00
Leifer
e5cecdee03 docs(contributin): added more info 2022-11-29 22:13:58 +01:00
Leifer
9351af16b7 docs(contributing): added more info 2022-11-29 22:04:46 +01:00
Leifer
ad8831a75a f 2022-11-29 22:00:05 +01:00
Leifer
c63018ff08 f 2022-11-29 21:55:39 +01:00
Leifer
131bce3898 . 2022-11-29 21:45:44 +01:00
Leifer
fff9316030 . 2022-11-29 21:44:04 +01:00
Leifer
2b3148dc3c update documenta 2022-11-29 21:43:11 +01:00
Leifer Mendez
13a4202f08 docs(contributing): added main readme 2022-11-29 11:33:33 +01:00
Leifer Mendez
f0df143aaf Merge pull request #125 from leifermendez/feature/monorepo
Feature/monorepo
2022-11-29 11:23:07 +01:00
Leifer Mendez
70a94ab2c6 fix(linter): update linter and commitlint 2022-11-29 10:47:12 +01:00
Leifer Mendez
e6d18d1a72 chore(release): 0.1.1 2022-11-29 10:36:23 +01:00
Leifer Mendez
46cd57fb36 chore(release): 0.1.0 2022-11-29 10:33:25 +01:00
Leifer Mendez
4ae389846d feat: (🎸) add onClick prop to component 2022-11-29 10:22:37 +01:00
Leifer Mendez
37d04e9e89 test: (💍) Is justa test!
nothing

BREAKING CHANGE: 🧨 NO
2022-11-29 10:17:32 +01:00
Leifer Mendez
39d141ca67 refactor: (💡) change emoji 2022-11-28 21:20:48 +01:00
Leifer Mendez
befcc169e0 refactor: 💡 pkgmanager
Upgrade Yarn v3
2022-11-28 21:19:00 +01:00
Leifer Mendez
6bbb9c1b81 docs(main documentation): es una mejora 2022-11-28 19:44:20 +01:00
Leifer Mendez
bb77afc4d2 test 2022-11-28 19:40:53 +01:00
43 changed files with 18975 additions and 5170 deletions

View File

@@ -4,5 +4,5 @@
"reporter": ["html"], "reporter": ["html"],
"report-dir": "./coverage", "report-dir": "./coverage",
"check-coverage": true, "check-coverage": true,
"lines": 95 "lines": 90
} }

9
.gitignore vendored
View File

@@ -14,12 +14,17 @@ mediaSend/*
.wwebjs_auth .wwebjs_auth
packages/cli/config.json packages/cli/config.json
config.json config.json
.yarnrc.yml
coverage/ coverage/
*.lcov *.lcov
log log
lib lib
tmp/ tmp/
.yarn/*
!.yarn/releases
!.yarn/plugins/@yarnpkg/plugin-postinstall.cjs
.fleet/ .fleet/
example-app/ example-app*/
qr.svg qr.svg
package-lock.json package-lock.json
yarn-error.log

4
.husky/commit-msg Executable file
View File

@@ -0,0 +1,4 @@
#!/usr/bin/env sh
. "$(dirname -- "$0")/_/husky.sh"
npx --no -- commitlint --edit

2
.husky/pre-commit Normal file → Executable file
View File

@@ -1,4 +1,4 @@
#!/usr/bin/env sh #!/usr/bin/env sh
. "$(dirname -- "$0")/_/husky.sh" . "$(dirname -- "$0")/_/husky.sh"
yarn run format:check && yarn run format:write && git add . yarn run fmt.staged

0
.husky/pre-push Normal file → Executable file
View File

3
.vscode/settings.json vendored Normal file
View File

@@ -0,0 +1,3 @@
{
"conventionalCommits.scopes": ["hook", "contributing", "cli"]
}

View File

@@ -0,0 +1,8 @@
/* eslint-disable */
module.exports = {
name: "@yarnpkg/plugin-postinstall",
factory: function (require) {
var plugin;(()=>{"use strict";var e={d:(t,n)=>{for(var o in n)e.o(n,o)&&!e.o(t,o)&&Object.defineProperty(t,o,{enumerable:!0,get:n[o]})},o:(e,t)=>Object.prototype.hasOwnProperty.call(e,t),r:e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})}},t={};e.r(t),e.d(t,{default:()=>s});const n=require("@yarnpkg/core"),o=require("clipanion"),a={postinstall:{description:"Postinstall hook that will always run in Yarn v2",type:n.SettingsType.STRING,default:""}},r=require("@yarnpkg/shell"),l=async e=>{if(e){console.log("Running postinstall command...");const t=await r.execute(e);if(0!==t)throw new Error("postinstall command failed with exit code "+t)}};var i=function(e,t,n,o){var a,r=arguments.length,l=r<3?t:null===o?o=Object.getOwnPropertyDescriptor(t,n):o;if("object"==typeof Reflect&&"function"==typeof Reflect.decorate)l=Reflect.decorate(e,t,n,o);else for(var i=e.length-1;i>=0;i--)(a=e[i])&&(l=(r<3?a(l):r>3?a(t,n,l):a(t,n))||l);return r>3&&l&&Object.defineProperty(t,n,l),l};class c extends o.Command{async execute(){const e=(await n.Configuration.find(this.context.cwd,this.context.plugins)).get("postinstall");await l(e)}}i([o.Command.Path("postinstall")],c.prototype,"execute",null);const s={configuration:a,commands:[c],hooks:{afterAllInstalled:async e=>{const t=e.configuration.get("postinstall");await l(t)}}};plugin=t})();
return plugin;
}
};

807
.yarn/releases/yarn-3.3.0.cjs vendored Normal file

File diff suppressed because one or more lines are too long

10
.yarnrc.yml Normal file
View File

@@ -0,0 +1,10 @@
nodeLinker: node-modules
npmPublishRegistry: 'https://registry.npmjs.org'
plugins:
- path: .yarn/plugins/@yarnpkg/plugin-postinstall.cjs
spec: 'https://raw.githubusercontent.com/gravitywelluk/yarn-plugin-postinstall/master/bundles/%40yarnpkg/plugin-postinstall.js'
yarnPath: .yarn/releases/yarn-3.3.0.cjs
postinstall: npx husky install

View File

@@ -1,3 +1,48 @@
# Changelog
All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
## 0.2.0-alpha.0 (2022-12-01)
### ⚠ BREAKING CHANGES
* 🧨 NO
### Features
* (🎸) add onClick prop to component ([4ae3898](https://github.com/leifermendez/bot-whatsapp/commit/4ae389846d38c133f6bb2129ae373eed39d9d08d))
### Bug Fixes
* **ci:** ci ([f55cfae](https://github.com/leifermendez/bot-whatsapp/commit/f55cfae6e4ccc1df949212999406680020d27f9c))
* **ci:** ci ([671c5b3](https://github.com/leifermendez/bot-whatsapp/commit/671c5b37f33360e8cb754625b8dd6e83bce9014d))
* **linter:** update linter and commitlint ([70a94ab](https://github.com/leifermendez/bot-whatsapp/commit/70a94ab2c6f8e4122780c77bc3a621944883e621))
* (💍) Is justa test! ([37d04e9](https://github.com/leifermendez/bot-whatsapp/commit/37d04e9e89d3f01fdc367654ba60fb11ab2614c4))
## 0.1.0 (2022-11-29)
### ⚠ BREAKING CHANGES
* 🧨 NO
### Features
* (🎸) add onClick prop to component ([4ae3898](https://github.com/leifermendez/bot-whatsapp/commit/4ae389846d38c133f6bb2129ae373eed39d9d08d))
### Bug Fixes
* **ci:** ci ([f55cfae](https://github.com/leifermendez/bot-whatsapp/commit/f55cfae6e4ccc1df949212999406680020d27f9c))
* **ci:** ci ([671c5b3](https://github.com/leifermendez/bot-whatsapp/commit/671c5b37f33360e8cb754625b8dd6e83bce9014d))
* (💍) Is justa test! ([37d04e9](https://github.com/leifermendez/bot-whatsapp/commit/37d04e9e89d3f01fdc367654ba60fb11ab2614c4))
#### Actualización 14 Ene 2022 #### Actualización 14 Ene 2022
- npm update - npm update

59
CONTRIBUTING.md Normal file
View File

@@ -0,0 +1,59 @@
# CONTRIBUTING
![](https://i.giphy.com/media/ntMt6TvalpstTIx7Ak/giphy.webp)
__Requerimientos:__
- Node v16 o superior __[descargar node](https://nodejs.org/es/download/)__
- __[Yarn](https://classic.yarnpkg.com/lang/en/docs/install/#windows-stable)__ como gestor de paquetes. En el link conseguiras las intrucciones para instalar yarn.
- __[VSCode](https://code.visualstudio.com/download)__ (recomendado): Editor de codigo con plugins
- __[Conventional Commits](https://marketplace.visualstudio.com/items?itemName=vivaxy.vscode-conventional-commits&ssr=false#overview)__ (plugin-vscode) este plugin te ayudara a crear commit semantico.
- Se usara la rama __dev__ *(https://github.com/leifermendez/bot-whatsapp/tree/dev)* como rama principal hasta que se haga oficialmente el lanzamiento de la V2
### 🚀 Iniciando
__Clonar repo rama dev__
```
git clone --branch dev https://github.com/leifermendez/bot-whatsapp
```
__Instalar dependencias__
```
cd bot-whatsapp
yarn install
```
__Compilar (build)__
Para compilar la aplicación es necesario ejecutar, eso te genera dentro de packages del monorepo un directorio `lib`
```
yarn build
```
__Example-app__
Se ejecuta el CLI (Command Line Interface) para ayudarte a crear un app-bot de ejemplo
```
yarn run cli
```
Abrir carpeta __example-app-base__ y ejecutar
```
cd example-app-base
npm i
npm run pre-copy
npm start
```
__Commit y Push__
El proyecto tiene implementado __[husky](https://typicode.github.io/husky/#/)__ es una herramienta que dispara unas acciones al momento de hacer commit y hacer push
__commit:__ Los commit son semanticos esto quiere decir que deben cumplir un standar al momento de escribirlos ejemplo ` feat(adapter): new adapter myqsl ` puede ver más info sobre esto __[aquí](https://github.com/conventional-changelog/commitlint/#what-is-commitlint)__
__push:__ Cada push ejecutar `yarn run test` el cual ejecuta los test internos que tienen que cumplir con __95% de cobertura__.
> Documento en constaten actualización....
------
- [Discord](https://link.codigoencasa.com/DISCORD)
- [Twitter](https://twitter.com/leifermendez)
- [Youtube](https://www.youtube.com/watch?v=5lEMCeWEJ8o&list=PL_WGMLcL4jzWPhdhcUyhbFU6bC0oJd2BR)
- [Telegram](https://t.me/leifermendez)

View File

@@ -1,3 +1,4 @@
```js ```js
const { const {

View File

@@ -1,6 +1,6 @@
[![Test / Coverage](https://github.com/leifermendez/bot-whatsapp/actions/workflows/ci.yml/badge.svg)](https://github.com/leifermendez/bot-whatsapp/actions/workflows/ci.yml) [![Test / Coverage](https://github.com/leifermendez/bot-whatsapp/actions/workflows/ci.yml/badge.svg)](https://github.com/leifermendez/bot-whatsapp/actions/workflows/ci.yml)
[![Commitizen friendly](https://img.shields.io/badge/commitizen-friendly-brightgreen.svg)](http://commitizen.github.io/cz-cli/) [![Commitizen friendly](https://img.shields.io/badge/commitizen-friendly-brightgreen.svg)](http://commitizen.github.io/cz-cli/)
--------
🦊 Documentación: [https://bot-whatsapp.pages.dev/](https://bot-whatsapp.pages.dev/) 🦊 Documentación: [https://bot-whatsapp.pages.dev/](https://bot-whatsapp.pages.dev/)
Video como hacer PR: https://youtu.be/Lxt8Acob6aU Video como hacer PR: https://youtu.be/Lxt8Acob6aU

View File

@@ -29,3 +29,4 @@
- [ ] Meta adapter - [ ] Meta adapter
### @bot-whatsapp/cli ### @bot-whatsapp/cli
- [ ] Hacer comando para crear `example-app`

91
changelog.config.js Normal file
View File

@@ -0,0 +1,91 @@
module.exports = {
disableEmoji: false,
format: '{type}{scope}: {emoji}{subject}',
list: [
'test',
'feat',
'fix',
'chore',
'docs',
'refactor',
'style',
'ci',
'perf',
],
maxMessageLength: 64,
minMessageLength: 3,
questions: [
'type',
'scope',
'subject',
'body',
'breaking',
'issues',
'lerna',
],
scopes: [],
types: {
chore: {
description: 'Build process or auxiliary tool changes',
emoji: '(🤖)',
value: 'chore',
},
ci: {
description: 'CI related changes',
emoji: '(🎡)',
value: 'ci',
},
docs: {
description: 'Documentation only changes',
emoji: '(✏️)',
value: 'docs',
},
feat: {
description: 'A new feature',
emoji: '(🎸)',
value: 'feat',
},
fix: {
description: 'A bug fix',
emoji: '(🐛)',
value: 'fix',
},
perf: {
description: 'A code change that improves performance',
emoji: '(⚡️)',
value: 'perf',
},
refactor: {
description:
'A code change that neither fixes a bug or adds a feature',
emoji: '(💡)',
value: 'refactor',
},
release: {
description: 'Create a release commit',
emoji: '(🏹)',
value: 'release',
},
style: {
description:
'Markup, white-space, formatting, missing semi-colons...',
emoji: '(💄)',
value: 'style',
},
test: {
description: 'Adding missing tests',
emoji: '(💍)',
value: 'test',
},
messages: {
type: "Select the type of change that you're committing:",
customScope: 'Select the scope this component affects:',
subject:
'Write a short, imperative mood description of the change:\n',
body: 'Provide a longer description of the change:\n ',
breaking: 'List any breaking changes:\n',
footer: 'Issues this commit closes, e.g #123:',
confirmCommit: 'The packages that this commit has affected\n',
},
},
}

1
commitlint.config.js Normal file
View File

@@ -0,0 +1 @@
module.exports = { extends: ['@commitlint/config-conventional'] }

View File

@@ -0,0 +1,8 @@
{
"banner.output": [
"/** \n",
"* NO TOCAR ESTE ARCHIVO: Es generado automaticamente, si sabes lo que haces adelante ;)\n",
"* de lo contrario mejor ir a la documentacion o al servidor de discord link.codigoencasa.com/DISCORD\n",
"*/"
]
}

View File

@@ -1,6 +1,6 @@
{ {
"name": "@bot-whatsapp/root", "name": "@bot-whatsapp/root",
"version": "0.0.1", "version": "0.2.0-alpha.0",
"description": "Bot de wahtsapp open source para MVP o pequeños negocios", "description": "Bot de wahtsapp open source para MVP o pequeños negocios",
"main": "app.js", "main": "app.js",
"private": true, "private": true,
@@ -12,10 +12,12 @@
"database:rollup": "rollup --config ./packages/database/rollup-database.config.js", "database:rollup": "rollup --config ./packages/database/rollup-database.config.js",
"format:check": "prettier --check ./packages", "format:check": "prettier --check ./packages",
"format:write": "prettier --write ./packages", "format:write": "prettier --write ./packages",
"fmt.staged": "pretty-quick --staged",
"lint:check": "eslint ./packages", "lint:check": "eslint ./packages",
"lint:fix": "eslint --fix ./packages", "lint:fix": "eslint --fix ./packages",
"build": "yarn run cli:rollup && yarn run bot:rollup && yarn run provider:rollup && yarn run database:rollup", "build": "yarn run cli:rollup && yarn run bot:rollup && yarn run provider:rollup && yarn run database:rollup",
"link.dist": "cd packages/bot && npm link && cd ../provider && npm link && cd ../cli && npm link", "link.dist": "cd packages/bot && npm link && cd ../provider && npm link && cd ../cli && npm link && cd ../database && npm link && cd ../provider && npm link",
"copy.lib": "node ./scripts/move.js",
"test.unit": "node ./node_modules/uvu/bin.js packages test", "test.unit": "node ./node_modules/uvu/bin.js packages test",
"test.coverage": "node ./node_modules/c8/bin/c8.js npm run test.unit", "test.coverage": "node ./node_modules/c8/bin/c8.js npm run test.unit",
"test": "npm run test.coverage", "test": "npm run test.coverage",
@@ -23,7 +25,9 @@
"dev:debug": "node --inspect ./example-app/app.js", "dev:debug": "node --inspect ./example-app/app.js",
"dev": "node ./example-app/app.js", "dev": "node ./example-app/app.js",
"prepare": "npx husky install", "prepare": "npx husky install",
"preinstall": "npx only-allow yarn" "preinstall": "npx only-allow yarn",
"postinstall": "npx prettier --write .",
"release": "standard-version"
}, },
"workspaces": [ "workspaces": [
"packages/bot", "packages/bot",
@@ -52,36 +56,36 @@
"repository": "https://github.com/leifermendez/bot-whatsapp", "repository": "https://github.com/leifermendez/bot-whatsapp",
"license": "ISC", "license": "ISC",
"devDependencies": { "devDependencies": {
"@commitlint/cli": "^17.3.0",
"@commitlint/config-conventional": "^17.3.0",
"@rollup/plugin-commonjs": "^23.0.2", "@rollup/plugin-commonjs": "^23.0.2",
"@rollup/plugin-json": "^5.0.1", "@rollup/plugin-json": "^5.0.1",
"@rollup/plugin-node-resolve": "^15.0.1", "@rollup/plugin-node-resolve": "^15.0.1",
"@rollup/plugin-replace": "^5.0.1", "@rollup/plugin-replace": "^5.0.1",
"c8": "^7.12.0", "c8": "^7.12.0",
"commitizen": "^4.2.5", "conventional-changelog": "^3.1.25",
"cross-env": "^7.0.3", "cross-env": "^7.0.3",
"cz-conventional-changelog": "^3.3.0",
"eslint": "^8.26.0", "eslint": "^8.26.0",
"eslint-config-prettier": "^8.5.0", "eslint-config-prettier": "^8.5.0",
"fs-extra": "^11.1.0",
"git-cz": "^4.9.0",
"husky": "^8.0.2", "husky": "^8.0.2",
"only-allow": "^1.1.1", "only-allow": "^1.1.1",
"prettier": "^2.7.1", "prettier": "^2.8.0",
"pretty-quick": "^3.1.3",
"prompts": "^2.4.2", "prompts": "^2.4.2",
"rimraf": "^3.0.2", "rimraf": "^3.0.2",
"rollup": "^3.2.3", "rollup": "^3.2.3",
"rollup-plugin-cleanup": "^3.2.1", "rollup-plugin-cleanup": "^3.2.1",
"rollup-plugin-copy": "^3.4.0", "rollup-plugin-copy": "^3.4.0",
"standard-version": "^9.5.0",
"uvu": "^0.5.6" "uvu": "^0.5.6"
}, },
"packageManager": "yarn@1.22.19", "packageManager": "yarn@3.3.0",
"engines": { "engines": {
"node": ">=16", "node": ">=16",
"npm": "please-use-yarn", "npm": "please-use-yarn",
"yarn": ">=1" "yarn": ">=3"
}, },
"author": "Leifer Mendez <leifer33@gmail.com>", "author": "Leifer Mendez <leifer33@gmail.com>"
"config": {
"commitizen": {
"path": "./node_modules/cz-conventional-changelog"
}
}
} }

View File

@@ -1,7 +1,7 @@
const CoreClass = require('./core/core.class') const CoreClass = require('./core/core.class')
const ProviderClass = require('./provider/provider.class') const ProviderClass = require('./provider/provider.class')
const FlowClass = require('./io/flow.class') const FlowClass = require('./io/flow.class')
const { addKeyword, addAnswer, toSerialize } = require('./io/methods') const { addKeyword, addAnswer, addChild, toSerialize } = require('./io/methods')
/** /**
* Crear instancia de clase Bot * Crear instancia de clase Bot
@@ -38,6 +38,7 @@ module.exports = {
createProvider, createProvider,
addKeyword, addKeyword,
addAnswer, addAnswer,
addChild,
toSerialize, toSerialize,
ProviderClass, ProviderClass,
CoreClass, CoreClass,

View File

@@ -10,6 +10,7 @@ const { toSerialize } = require('./toSerialize')
const addAnswer = const addAnswer =
(inCtx) => (inCtx) =>
(answer, options, cb = null, nested = []) => { (answer, options, cb = null, nested = []) => {
answer = Array.isArray(answer) ? answer.join('\n') : answer
/** /**
* Todas las opciones referentes a el mensaje en concreto options:{} * Todas las opciones referentes a el mensaje en concreto options:{}
* @returns * @returns

View File

@@ -0,0 +1,15 @@
const { toSerialize } = require('./toSerialize')
/**
* @deprecate
* @param answer string
* @param options {media:string, buttons:[], capture:true default false}
* @returns
*/
const addChild = (flowIn = null) => {
if (!flowIn?.toJson) {
throw new Error('DEBE SER UN FLOW CON toJSON()')
}
return toSerialize(flowIn.toJson())
}
module.exports = { addChild }

View File

@@ -1,7 +1,8 @@
const { addAnswer } = require('./addAnswer') const { addAnswer } = require('./addAnswer')
const { addKeyword } = require('./addKeyword') const { addKeyword } = require('./addKeyword')
const { addChild } = require('./addChild')
const { toSerialize } = require('./toSerialize') const { toSerialize } = require('./toSerialize')
const { toCtx } = require('./toCtx') const { toCtx } = require('./toCtx')
const { toJson } = require('./toJson') const { toJson } = require('./toJson')
module.exports = { addAnswer, addKeyword, toCtx, toJson, toSerialize } module.exports = { addAnswer, addKeyword, addChild, toCtx, toJson, toSerialize }

View File

@@ -1,3 +1,4 @@
const banner = require('../../config/banner.rollup.json')
const commonjs = require('@rollup/plugin-commonjs') const commonjs = require('@rollup/plugin-commonjs')
const { nodeResolve } = require('@rollup/plugin-node-resolve') const { nodeResolve } = require('@rollup/plugin-node-resolve')
const { join } = require('path') const { join } = require('path')
@@ -7,6 +8,7 @@ const PATH = join(__dirname, 'lib', 'bundle.bot.cjs')
module.exports = { module.exports = {
input: join(__dirname, 'index.js'), input: join(__dirname, 'index.js'),
output: { output: {
banner: banner['banner.output'].join(''),
file: PATH, file: PATH,
format: 'cjs', format: 'cjs',
}, },

View File

@@ -22,6 +22,15 @@ test('Debere probar las propeidades array', () => {
assert.is(MAIN_CTX.ctx.keyword, ARRANGE.keyword) assert.is(MAIN_CTX.ctx.keyword, ARRANGE.keyword)
}) })
test('Debere probar las propeidades array en answer', () => {
const ARRANGE = {
keyword: ['hola!', 'ole'],
}
const MAIN_CTX = addKeyword(ARRANGE.keyword).addAnswer(['hola', 'chao'])
assert.is(MAIN_CTX.ctx.keyword, ARRANGE.keyword)
})
test('Debere probar toSerialize', () => { test('Debere probar toSerialize', () => {
const ARRANGE = { const ARRANGE = {
keyword: ['hola!', 'ole'], keyword: ['hola!', 'ole'],

0
packages/cli/bin/cli.js Normal file → Executable file
View File

View File

@@ -0,0 +1,21 @@
const fs = require('fs-extra')
/**
* Copy files
*/
const copyFiles = async (from, to) => {
try {
await fs.copy(from, to)
console.log('success!')
} catch (err) {
console.error(err)
}
}
const copyBaseApp = async () => {
const BASEP_APP_PATH_FROM = `${process.cwd()}/starters/apps/base`
const BASEP_APP_PATH_TO = `${process.cwd()}/example-app-base`
await copyFiles(BASEP_APP_PATH_FROM, BASEP_APP_PATH_TO)
}
module.exports = { copyBaseApp }

View File

@@ -2,6 +2,7 @@ const prompts = require('prompts')
const { yellow, red } = require('kleur') const { yellow, red } = require('kleur')
const { installAll } = require('../install') const { installAll } = require('../install')
const { cleanSession } = require('../clean') const { cleanSession } = require('../clean')
const { copyBaseApp } = require('../create-app')
const { checkNodeVersion, checkOs } = require('../check') const { checkNodeVersion, checkOs } = require('../check')
const { jsonConfig } = require('../configuration') const { jsonConfig } = require('../configuration')
@@ -9,10 +10,16 @@ const startInteractive = async () => {
const questions = [ const questions = [
{ {
type: 'text', type: 'text',
name: 'dependencies', name: 'exampeOpt',
message: message:
'Quieres actualizar las librerias "whatsapp-web.js"? (Y/n)', 'Quieres crear una app de ejemplo "example-app-example"? (Y/n)',
}, },
// {
// type: 'text',
// name: 'dependencies',
// message:
// 'Quieres actualizar las librerias "whatsapp-web.js"? (Y/n)',
// },
{ {
type: 'text', type: 'text',
name: 'cleanTmp', name: 'cleanTmp',
@@ -57,11 +64,12 @@ const startInteractive = async () => {
const { const {
dependencies = '', dependencies = '',
cleanTmp = '', cleanTmp = '',
exampeOpt = '',
providerDb = [], providerDb = [],
providerWs = [], providerWs = [],
} = response } = response
/** /**
* Question #1 * Question
* @returns * @returns
*/ */
const installOrUdpateDep = async () => { const installOrUdpateDep = async () => {
@@ -75,7 +83,7 @@ const startInteractive = async () => {
} }
/** /**
* Question #2 * Question
* @returns * @returns
*/ */
const cleanAllSession = async () => { const cleanAllSession = async () => {
@@ -88,6 +96,16 @@ const startInteractive = async () => {
} }
} }
const createApp = async () => {
const answer = exampeOpt.toLowerCase() || 'n'
if (answer.includes('n')) return true
if (answer.includes('y')) {
await copyBaseApp()
return true
}
}
const vendorProvider = async () => { const vendorProvider = async () => {
if (!providerWs.length) { if (!providerWs.length) {
console.log( console.log(
@@ -117,6 +135,7 @@ const startInteractive = async () => {
} }
} }
await createApp()
await installOrUdpateDep() await installOrUdpateDep()
await cleanAllSession() await cleanAllSession()
await vendorProvider() await vendorProvider()

View File

@@ -1,3 +1,4 @@
const banner = require('../../config/banner.rollup.json')
const commonjs = require('@rollup/plugin-commonjs') const commonjs = require('@rollup/plugin-commonjs')
const { nodeResolve } = require('@rollup/plugin-node-resolve') const { nodeResolve } = require('@rollup/plugin-node-resolve')
const { join } = require('path') const { join } = require('path')
@@ -7,6 +8,7 @@ const PATH = join(__dirname, 'lib', 'cli', 'bundle.cli.cjs')
module.exports = { module.exports = {
input: join(__dirname, 'index.js'), input: join(__dirname, 'index.js'),
output: { output: {
banner: banner['banner.output'].join(''),
file: PATH, file: PATH,
format: 'cjs', format: 'cjs',
}, },

View File

@@ -25,6 +25,28 @@ const main = async () => {
} }
``` ```
#### CTX
```json
{
ref: 'ans_7d9981e5-5019-422c-a19a-565cbb021391',
keyword: 'ans_cfdad31b-ff6d-475f-873a-4ed6f8a79a43',
answer: 'Esperando respuesta...',
options: {
media: null,
buttons: [],
capture: true,
child: null,
nested: [Array],
keyword: {},
callback: true
},
refSerialize: '81f18f563fd26a6c6d12c62aed98095f',
from: 'NUMERO_PERSONA_QUE_ESCRIBE'
}
```
#### Video #### Video
> Video explicando como debes de agregar nuevos adaptadores > Video explicando como debes de agregar nuevos adaptadores

View File

@@ -7,13 +7,14 @@
"keywords": [], "keywords": [],
"author": "", "author": "",
"license": "ISC", "license": "ISC",
"devDependencies": {},
"dependencies": { "dependencies": {
"dotenv": "^16.0.3", "dotenv": "^16.0.3",
"mongodb": "^4.11.0" "mongodb": "^4.11.0",
"mysql2": "^2.3.3"
}, },
"exports": { "exports": {
"./mock": "./lib/mock/index.cjs", "./mock": "./lib/mock/index.cjs",
"./mongo": "./lib/mongo/index.cjs" "./mongo": "./lib/mongo/index.cjs",
"./mysql": "./lib/mysql/index.cjs"
} }
} }

View File

@@ -1,3 +1,4 @@
const banner = require('../../config/banner.rollup.json')
const commonjs = require('@rollup/plugin-commonjs') const commonjs = require('@rollup/plugin-commonjs')
const { join } = require('path') const { join } = require('path')
@@ -5,6 +6,7 @@ module.exports = [
{ {
input: join(__dirname, 'src', 'mock', 'index.js'), input: join(__dirname, 'src', 'mock', 'index.js'),
output: { output: {
banner: banner['banner.output'].join(''),
file: join(__dirname, 'lib', 'mock', 'index.cjs'), file: join(__dirname, 'lib', 'mock', 'index.cjs'),
format: 'cjs', format: 'cjs',
}, },
@@ -13,9 +15,19 @@ module.exports = [
{ {
input: join(__dirname, 'src', 'mongo', 'index.js'), input: join(__dirname, 'src', 'mongo', 'index.js'),
output: { output: {
banner: banner['banner.output'].join(''),
file: join(__dirname, 'lib', 'mongo', 'index.cjs'), file: join(__dirname, 'lib', 'mongo', 'index.cjs'),
format: 'cjs', format: 'cjs',
}, },
plugins: [commonjs()], plugins: [commonjs()],
}, },
{
input: join(__dirname, 'src', 'mysql', 'index.js'),
output: {
banner: banner['banner.output'].join(''),
file: join(__dirname, 'lib', 'mysql', 'index.cjs'),
format: 'cjs',
},
plugins: [commonjs()],
},
] ]

View File

@@ -1,15 +1,20 @@
/**
* Si necesitas saber que trae el "ctx"
* Puedes ver el README.md dentro packages/database
*/
class MockDatabase { class MockDatabase {
db
listHistory = [] listHistory = []
constructor() { constructor() {}
/**
* Se debe cargar listHistory con historial de mensajes getPrevByNumber = (from) => {
* para que se pueda continuar el flow const history = this.listHistory.slice().reverse()
*/ return history.find((a) => a.from === from)
} }
save = (ctx) => { save = (ctx) => {
console.log('Guardando DB...', ctx)
this.listHistory.push(ctx) this.listHistory.push(ctx)
} }
} }

View File

@@ -0,0 +1,76 @@
require('dotenv').config()
const mysql = require('mysql2')
const DB_NAME = process.env.DB_NAME || 'db_bot'
const DB_HOST = process.env.DB_HOST || 'localhost'
const DB_USER = process.env.DB_USER || 'root'
class MyslAdapter {
db
listHistory = []
constructor() {
this.init().then()
}
async init() {
this.db = mysql.createConnection({
host: DB_HOST,
user: DB_USER,
database: DB_NAME,
})
await this.db.connect((error) => {
if (!error) {
console.log(`Solicitud de conexión a base de datos exitosa`)
}
if (error) {
console.log(`Solicitud de conexión fallida ${error.stack}`)
}
})
}
getPrevByNumber = (from) =>
new Promise((resolve, reject) => {
const sql = `SELECT * FROM history WHERE phone=${from} ORDER BY id DESC`
this.db.query(sql, (error, rows) => {
if (error) {
reject(error)
}
if (rows.length) {
const [row] = rows
row.options = JSON.parse(row.options)
resolve(row)
}
if (!rows.length) {
resolve(null)
}
})
})
save = (ctx) => {
const values = [
[
ctx.ref,
ctx.keyword,
ctx.answer,
ctx.refSerialize,
ctx.from,
JSON.stringify(ctx.options),
],
]
const sql =
'INSERT INTO history (ref, keyword, answer, refSerialize, phone, options ) values ?'
this.db.query(sql, [values], (err) => {
if (err) throw err
console.log('Guardado en DB...', values)
})
this.listHistory.push(ctx)
}
}
module.exports = MyslAdapter

View File

@@ -1,3 +1,4 @@
const banner = require('../../config/banner.rollup.json')
const { join } = require('path') const { join } = require('path')
const commonjs = require('@rollup/plugin-commonjs') const commonjs = require('@rollup/plugin-commonjs')
@@ -5,6 +6,7 @@ module.exports = [
{ {
input: join(__dirname, 'src', 'web-whatsapp', 'index.js'), input: join(__dirname, 'src', 'web-whatsapp', 'index.js'),
output: { output: {
banner: banner['banner.output'].join(''),
file: join(__dirname, 'lib', 'web-whatsapp', 'index.cjs'), file: join(__dirname, 'lib', 'web-whatsapp', 'index.cjs'),
format: 'cjs', format: 'cjs',
}, },
@@ -13,6 +15,7 @@ module.exports = [
{ {
input: join(__dirname, 'src', 'twilio', 'index.js'), input: join(__dirname, 'src', 'twilio', 'index.js'),
output: { output: {
banner: banner['banner.output'].join(''),
file: join(__dirname, 'lib', 'twilio', 'index.cjs'), file: join(__dirname, 'lib', 'twilio', 'index.cjs'),
format: 'cjs', format: 'cjs',
}, },
@@ -21,6 +24,7 @@ module.exports = [
{ {
input: join(__dirname, 'src', 'mock', 'index.js'), input: join(__dirname, 'src', 'mock', 'index.js'),
output: { output: {
banner: banner['banner.output'].join(''),
file: join(__dirname, 'lib', 'mock', 'index.cjs'), file: join(__dirname, 'lib', 'mock', 'index.cjs'),
format: 'cjs', format: 'cjs',
}, },

17
scripts/move.js Normal file
View File

@@ -0,0 +1,17 @@
const fs = require('fs-extra')
const PACKAGES_PATH = `${process.cwd()}/packages`
const NAME_PREFIX = `@bot-whatsapp`
const [, , appDir] = process.argv || []
const copyLibPkg = async (pkgName, to) => {
const FROM = `${PACKAGES_PATH}/${pkgName}`
const TO = `${process.cwd()}/${to}/node_modules/${NAME_PREFIX}/${pkgName}`
await fs.copy(FROM, TO)
}
Promise.all([
copyLibPkg('bot', appDir),
copyLibPkg('database', appDir),
copyLibPkg('provider', appDir),
]).then(() => console.log('Todas las lib copiadas'))

View File

@@ -0,0 +1,12 @@
### BASE APP
Este bot contiene un flujo basico en el cual una persona (cliente) escribe **"hola"** y el bot responde.
- Bienvenido a mi tienda
- Como puedo ayudarte?
- Tengo: Zapatos Bolsos etc..
------
- [Discord](https://link.codigoencasa.com/DISCORD)
- [Twitter](https://twitter.com/leifermendez)
- [Youtube](https://www.youtube.com/watch?v=5lEMCeWEJ8o&list=PL_WGMLcL4jzWPhdhcUyhbFU6bC0oJd2BR)
- [Telegram](https://t.me/leifermendez)

33
starters/apps/base/app.js Normal file
View File

@@ -0,0 +1,33 @@
const {
createBot,
createProvider,
createFlow,
addKeyword,
} = require('@bot-whatsapp/bot')
/**
* ATENCION: Si vas a usar el provider whatsapp-web.js
* recuerda ejecutar npm i whatsapp-web.js@latest
*/
const WebWhatsappProvider = require('@bot-whatsapp/provider/web-whatsapp')
const MockAdapter = require('@bot-whatsapp/database/mock')
const flowPrincipal = addKeyword(['hola', 'ole', 'HOLA'])
.addAnswer('Bienvenido a mi tienda')
.addAnswer('Como puedo ayudarte?')
.addAnswer(['Tengo:', 'Zapatos', 'Bolsos', 'etc..'])
const main = async () => {
const adapterDB = new MockAdapter()
const adapterFlow = createFlow([flowPrincipal])
const adapterProvider = createProvider(WebWhatsappProvider)
createBot({
flow: adapterFlow,
provider: adapterProvider,
database: adapterDB,
})
}
main()

View File

@@ -0,0 +1,16 @@
{
"name": "bot-whatsapp-base",
"version": "1.0.0",
"description": "",
"main": "app.js",
"scripts": {
"pre-copy": "cd .. && yarn run copy.lib example-app-base",
"start": "node app.js"
},
"keywords": [],
"dependencies": {
"whatsapp-web.js": "^1.18.3"
},
"author": "",
"license": "ISC"
}

View File

@@ -0,0 +1,12 @@
### BASIC APP
Este bot contiene un flujo basico en el cual una persona (cliente) escribe **"hola"** y el bot responde.
![](https://i.imgur.com/0z0G91w.png)
------
- [Discord](https://link.codigoencasa.com/DISCORD)
- [Twitter](https://twitter.com/leifermendez)
- [Youtube](https://www.youtube.com/watch?v=5lEMCeWEJ8o&list=PL_WGMLcL4jzWPhdhcUyhbFU6bC0oJd2BR)
- [Telegram](https://t.me/leifermendez)

View File

@@ -0,0 +1,60 @@
const {
createBot,
createProvider,
createFlow,
addKeyword,
addChild,
} = require('@bot-whatsapp/bot')
/**
* ATENCION: Si vas a usar el provider whatsapp-web.js
* recuerda ejecutar npm i whatsapp-web.js@latest
*/
const WebWhatsappProvider = require('@bot-whatsapp/provider/web-whatsapp')
const MockAdapter = require('@bot-whatsapp/database/mock')
/**
* Declarando flujo hijo
*/
const flowZapatos = addKeyword(['zapatos', 'ZAPATOS'])
.addAnswer('🤯 Veo que elegiste zapatos')
.addAnswer('Tengo muchos zapatos...bla bla')
const flowBolsos = addKeyword(['bolsos', 'BOLSOS'])
.addAnswer('🙌 Veo que elegiste bolsos')
.addAnswer('Tengo muchos bolsos...bla bla')
/**
* Declarando flujo principal
*/
const flowPrincipal = addKeyword(['hola', 'ole', 'HOLA'])
.addAnswer('Bienvenido a mi tienda')
.addAnswer('Como puedo ayudarte?')
.addAnswer(['Tengo:', 'Zapatos', 'Bolsos', 'etc..'])
.addAnswer('Escribe zapatos o bolsos')
.addAnswer(
'esperando respuesta...',
{ capture: true },
(ctx) => {
console.log('Aqui puedes ver más info del usuario...')
console.log('Puedes enviar un mail, hook, etc..')
console.log(ctx)
},
[...addChild(flowBolsos), ...addChild(flowZapatos)]
)
const main = async () => {
const adapterDB = new MockAdapter()
const adapterFlow = createFlow([flowPrincipal])
const adapterProvider = createProvider(WebWhatsappProvider)
createBot({
flow: adapterFlow,
provider: adapterProvider,
database: adapterDB,
})
}
main()

View File

@@ -0,0 +1,16 @@
{
"name": "bot-whatsapp-basic",
"version": "1.0.0",
"description": "",
"main": "app.js",
"scripts": {
"pre-copy": "cd .. && yarn run copy.lib example-app-basic",
"start": "node app.js"
},
"keywords": [],
"dependencies": {
"whatsapp-web.js": "^1.18.3"
},
"author": "",
"license": "ISC"
}

8944
yarn-error.log Normal file

File diff suppressed because it is too large Load Diff

13729
yarn.lock

File diff suppressed because it is too large Load Diff