mirror of
https://github.com/cheveguerra/bot-whatsapp.git
synced 2026-04-20 20:49:15 +00:00
Compare commits
1258 Commits
add-code-o
...
dev
| Author | SHA1 | Date | |
|---|---|---|---|
| 8067c91904 | |||
|
|
420be044f1 | ||
|
|
7e599bf9d0 | ||
|
|
8d46862b0d | ||
|
|
4d0d455865 | ||
|
|
565f995c24 | ||
| ac7c56694a | |||
|
|
b3680bd363 | ||
|
|
eb79505c19 | ||
|
|
1c602ff955 | ||
|
|
ce066c3922 | ||
|
|
647d6afd4b | ||
|
|
e26040f4ce | ||
|
|
9a7f113d66 | ||
|
|
1eb0403529 | ||
|
|
2c20d47c53 | ||
|
|
b158e50824 | ||
|
|
104cde012c | ||
|
|
8609c30c89 | ||
|
|
856165e7a5 | ||
|
|
bd9093956a | ||
|
|
2f07d86fb9 | ||
|
|
64cea7f050 | ||
|
|
405d73af04 | ||
|
|
bea38cff3b | ||
|
|
a147677a26 | ||
|
|
2b2b4ef2ee | ||
|
|
c7de860803 | ||
|
|
6933fed395 | ||
|
|
a46a9efd8d | ||
|
|
b343204040 | ||
|
|
ba2ed54405 | ||
|
|
b558afb3f9 | ||
|
|
c9a6350f42 | ||
|
|
6f7808c167 | ||
|
|
b238752df1 | ||
|
|
a88b01e69f | ||
|
|
31a42ddcad | ||
|
|
dda59fa93b | ||
| d2bcf4cc0e | |||
|
|
14ee9eea89 | ||
|
|
cfc5309e52 | ||
|
|
af803415e3 | ||
|
|
c0ece6feb2 | ||
|
|
2278149295 | ||
|
|
2d44a002ff | ||
|
|
652d0ef2c3 | ||
|
|
9a63fa54e1 | ||
|
|
0b6729c6ad | ||
|
|
309dd1da99 | ||
|
|
93bee15ab8 | ||
|
|
276fc0cf88 | ||
| 9af15d5389 | |||
|
|
5d38cd377c | ||
|
|
176f337f18 | ||
|
|
76ba717927 | ||
|
|
2ecc41f3f0 | ||
|
|
99b9b17f52 | ||
|
|
cd082f2350 | ||
|
|
1afc3ba182 | ||
|
|
8b79e4e616 | ||
|
|
c9831d202a | ||
|
|
9311aa0a65 | ||
|
|
e34560c77d | ||
|
|
d220f2c622 | ||
|
|
c0113ca492 | ||
|
|
aef52d2694 | ||
|
|
f76932021c | ||
| 5b1f070a53 | |||
|
|
23e09efaec | ||
|
|
96c2bffd09 | ||
|
|
c3a4162ff6 | ||
|
|
186c48d884 | ||
|
|
bb31045a95 | ||
|
|
74fb3b864d | ||
|
|
88c05c12a4 | ||
|
|
5260a7eb47 | ||
|
|
9e1698b729 | ||
|
|
811618b256 | ||
|
|
331d2a309c | ||
|
|
c655c48009 | ||
|
|
ebd30ea1d7 | ||
|
|
0f8a33cc0c | ||
|
|
85b2d13290 | ||
|
|
318ac3adec | ||
|
|
0a69b8d9b5 | ||
|
|
f434d6a101 | ||
|
|
753d80b93e | ||
|
|
c07e148dfd | ||
|
|
3f30a1cb51 | ||
|
|
9b679192db | ||
|
|
f7d90efc2f | ||
|
|
de5de37d65 | ||
|
|
0aa793e08c | ||
|
|
eeb4bb305e | ||
|
|
012d43847c | ||
|
|
87a4203cd5 | ||
|
|
f6114affad | ||
|
|
15b64185cb | ||
|
|
b655ae449e | ||
|
|
1c66f178a5 | ||
|
|
f201c5097b | ||
|
|
b33e34692d | ||
|
|
8f967578c6 | ||
|
|
fd2847aea0 | ||
|
|
f95331d3dc | ||
|
|
791ab8e970 | ||
|
|
880c729199 | ||
|
|
8da4b204b4 | ||
|
|
3fdd49ff86 | ||
|
|
e22780d3fa | ||
|
|
0ad4c58457 | ||
|
|
f8c7184487 | ||
|
|
b2afa45352 | ||
|
|
dcb0566d2b | ||
|
|
cbe438b778 | ||
|
|
6ff1a3a980 | ||
|
|
dceb13f4f5 | ||
|
|
bbbdb1c206 | ||
|
|
bd7d150c04 | ||
|
|
9dd7c02b6a | ||
|
|
11a74b8bea | ||
|
|
e7a8e85ead | ||
|
|
6bfbae7b94 | ||
|
|
d5d7f9dfee | ||
|
|
0a23d2c761 | ||
|
|
b2feaea588 | ||
|
|
01c7db8fe7 | ||
| d87dc3c188 | |||
|
|
7e2bf22d63 | ||
|
|
a4b610e21f | ||
|
|
120520df50 | ||
|
|
1551aafd54 | ||
|
|
eca876db9c | ||
|
|
96a387ed50 | ||
|
|
558013b2b4 | ||
|
|
58df7ddc71 | ||
|
|
e25f3abf1c | ||
|
|
0f6b82a9ab | ||
|
|
8f505bdb2b | ||
|
|
24220822f4 | ||
|
|
14d1a61fa2 | ||
|
|
943fe8698c | ||
|
|
371b403456 | ||
|
|
e19c3a25a4 | ||
|
|
eab39e4ac0 | ||
|
|
77145bcc54 | ||
|
|
8f241834e8 | ||
|
|
5174c6b3bb | ||
| d9639c8e8d | |||
|
|
f3ed6da4ba | ||
|
|
0dc839e531 | ||
|
|
7475d6f8fd | ||
|
|
a6a33ac7b7 | ||
|
|
71c969f3e9 | ||
|
|
cebfed0382 | ||
|
|
e89ad450a1 | ||
|
|
ac39ac831c | ||
|
|
0af74602f5 | ||
|
|
767e0764d6 | ||
|
|
b2a3343f82 | ||
|
|
5ddd885554 | ||
|
|
7078dc4c93 | ||
|
|
c7e829e954 | ||
|
|
5a81a77802 | ||
|
|
588411653e | ||
|
|
3c4b1c0fc4 | ||
|
|
0f06fd3e80 | ||
|
|
79cc31a96f | ||
|
|
7067b4a80b | ||
|
|
aa7e4239ae | ||
|
|
877252bd4a | ||
|
|
f5a7de3a00 | ||
| 10ab7e671f | |||
|
|
c792d47344 | ||
|
|
f6130cf0b9 | ||
|
|
9f9d833925 | ||
| 2631e4dcb4 | |||
|
|
02c0d8af76 | ||
|
|
880d5323e8 | ||
|
|
9bb33582fb | ||
|
|
c8ff84e9cd | ||
|
|
e942bd1d5e | ||
|
|
7e557bdd30 | ||
|
|
fcd1a63676 | ||
|
|
4ade5f02a7 | ||
|
|
05c834d6b1 | ||
|
|
234cc3ffb6 | ||
|
|
e321f35d86 | ||
|
|
5edd755491 | ||
|
|
10748a46a4 | ||
|
|
5b3136999b | ||
|
|
cb047cca8e | ||
|
|
4b8c09633e | ||
|
|
7cabc53eed | ||
|
|
40c34dd7a5 | ||
|
|
7e25dcaa93 | ||
|
|
bfa622fad0 | ||
|
|
e88141077f | ||
|
|
d743fdcfe3 | ||
|
|
1261580004 | ||
| 714718c0fc | |||
|
|
a14c67ad45 | ||
|
|
c5af1f8107 | ||
|
|
01a4edb7e3 | ||
|
|
98793d0cfc | ||
|
|
168b81315b | ||
|
|
8520b09dac | ||
|
|
adb0e49029 | ||
|
|
464dd44ce2 | ||
|
|
e3c94a49d7 | ||
|
|
a29b9d4e1f | ||
|
|
e036817e84 | ||
|
|
18ef4e9d72 | ||
|
|
fe129f62fe | ||
|
|
bb6ed4a084 | ||
|
|
9234cf1c5d | ||
|
|
a118bbbf7f | ||
|
|
f54dea52b0 | ||
|
|
72e0a91050 | ||
|
|
70dd4d73e8 | ||
|
|
ecf0eef928 | ||
|
|
345f256a1b | ||
|
|
3648757fa0 | ||
|
|
32f6a70f8f | ||
|
|
8c825e7f6b | ||
|
|
0c0f4375b8 | ||
|
|
039ce5dd7c | ||
|
|
5e879188b8 | ||
|
|
21a7270281 | ||
|
|
82a99b2c80 | ||
|
|
cc19974579 | ||
|
|
56fcb8fb72 | ||
|
|
f36cff1eef | ||
|
|
09fd0dd2e1 | ||
|
|
b393c11af6 | ||
|
|
6683715ad6 | ||
|
|
8cbfd560a3 | ||
| ea5a806392 | |||
| 6ecf97678f | |||
|
|
e72794a296 | ||
|
|
13e0530c01 | ||
|
|
e5ee70f088 | ||
|
|
bb1f60c6e3 | ||
|
|
b9a1703b20 | ||
|
|
f274379c53 | ||
|
|
d18efc88a9 | ||
|
|
03b5b441de | ||
|
|
06862abece | ||
|
|
d25307dd2d | ||
|
|
04c5209cac | ||
|
|
5362fcec7a | ||
|
|
2f8d5a3fb3 | ||
|
|
4f32e6f2c2 | ||
|
|
553d7fcba9 | ||
| d483b91359 | |||
|
|
6aed100d96 | ||
|
|
c8335c37c2 | ||
|
|
4a7c2fe553 | ||
|
|
a02e373cd9 | ||
| a9755da8c9 | |||
| 46a9fa6793 | |||
|
|
4ee1136115 | ||
|
|
b60db404a9 | ||
|
|
44efbd8518 | ||
|
|
8357eda573 | ||
|
|
05dbf2f240 | ||
|
|
b65c40e59b | ||
|
|
d8bb1418dd | ||
|
|
8f27d34081 | ||
|
|
ee30df73b8 | ||
|
|
8a1bdc5878 | ||
|
|
f742caa9a4 | ||
|
|
a1ce98bdc8 | ||
|
|
230981e267 | ||
|
|
7242ab92f7 | ||
|
|
d3c937579a | ||
| 1302491869 | |||
|
|
45cbc09b0e | ||
|
|
aaec075140 | ||
|
|
42b98801db | ||
|
|
b681be3f11 | ||
|
|
e442d05aa4 | ||
|
|
b682b0b6ba | ||
|
|
96b6430efe | ||
|
|
9723e9d908 | ||
|
|
1bea3401dc | ||
|
|
0f9e57df37 | ||
|
|
49b9b77543 | ||
|
|
804ea2b258 | ||
|
|
939cea03cc | ||
|
|
89cf909888 | ||
|
|
f323184b34 | ||
|
|
65ad00e4ec | ||
|
|
c4158cbac6 | ||
|
|
5a06d2cf88 | ||
|
|
675046cc58 | ||
| f430380b4f | |||
| 8a1c9f3b0e | |||
| eceb170df0 | |||
|
|
ef03478683 | ||
|
|
fb3e2913e3 | ||
|
|
9fb34cc9e0 | ||
|
|
3d2c3e1452 | ||
|
|
83e262b818 | ||
|
|
389169fc58 | ||
|
|
f80a051fa9 | ||
|
|
9688ec1844 | ||
|
|
288894ee8e | ||
|
|
f3151850d5 | ||
|
|
49e9c70d7c | ||
|
|
6a156183ad | ||
|
|
f029c7317a | ||
|
|
fea6276b89 | ||
|
|
4b8c1d0ec6 | ||
|
|
552a425cc0 | ||
|
|
7566059073 | ||
|
|
b321b03b2a | ||
|
|
edf5fdcbdc | ||
|
|
eb1fcbff22 | ||
|
|
68a6e4b241 | ||
|
|
837bbf3c36 | ||
|
|
c8c5a03bad | ||
|
|
353e25c204 | ||
|
|
be6c609570 | ||
|
|
c6f58e913d | ||
|
|
e2fec5d521 | ||
|
|
681ec2e1ab | ||
|
|
0dc5f99089 | ||
|
|
5c880eddf1 | ||
|
|
57fc150331 | ||
|
|
9704bc0f02 | ||
|
|
7d65fee978 | ||
|
|
faa3a21ffe | ||
|
|
84f689cee1 | ||
|
|
5175f66dd0 | ||
|
|
0af9ef2dae | ||
|
|
787cdc69a0 | ||
|
|
f8ceb01eca | ||
|
|
d52951adf9 | ||
|
|
c621d0ee24 | ||
|
|
f047ba6e71 | ||
|
|
401c012e00 | ||
|
|
39859f819c | ||
|
|
5b3e7cefde | ||
|
|
8436d7b0d3 | ||
|
|
135961daf0 | ||
|
|
375a997f25 | ||
|
|
625ca405fc | ||
|
|
8e3413ca80 | ||
|
|
09407401eb | ||
|
|
1fa17dc601 | ||
|
|
73f3e6245d | ||
|
|
afb127278a | ||
|
|
15bfb19853 | ||
|
|
d90a42b962 | ||
|
|
0619641f82 | ||
|
|
d00547866a | ||
|
|
ca978867ff | ||
|
|
af44ed1268 | ||
|
|
2a416bf751 | ||
|
|
2fc274725d | ||
|
|
ecde23fdea | ||
|
|
a73ff9c217 | ||
|
|
16e4575777 | ||
|
|
74a1301342 | ||
|
|
210c013290 | ||
|
|
bfb76c87a1 | ||
|
|
8f2eb6d4f2 | ||
|
|
44dfe8ed1a | ||
|
|
63ecac6f2b | ||
|
|
ac72c5e4e7 | ||
|
|
50d64596c7 | ||
|
|
a7801d1fa2 | ||
|
|
74e6aae631 | ||
|
|
c913d5ee5c | ||
|
|
eafd27c25a | ||
|
|
4289b7c3dd | ||
|
|
c793890c50 | ||
|
|
48a87fff26 | ||
|
|
98bff0285d | ||
|
|
f5b8e42608 | ||
|
|
6469c83552 | ||
|
|
dba33359a0 | ||
|
|
14f3893727 | ||
|
|
a6607f1896 | ||
|
|
9669cc0807 | ||
|
|
1ee05d925a | ||
|
|
3e18bc1f57 | ||
|
|
5872d860c9 | ||
|
|
caa2a42933 | ||
|
|
6b407a1d72 | ||
|
|
da67b75d98 | ||
|
|
28b5739f3f | ||
|
|
2d9da1e584 | ||
|
|
11bcb3f2c8 | ||
|
|
7731f4149e | ||
|
|
ae645643d5 | ||
|
|
9960227006 | ||
|
|
c0bbf5c48f | ||
|
|
512207207c | ||
|
|
200dd6b187 | ||
|
|
cad73dca99 | ||
|
|
2114f997db | ||
|
|
c5d4d82538 | ||
|
|
fdab223786 | ||
|
|
f4d1ae3199 | ||
|
|
527eac4962 | ||
|
|
8ef6d6b560 | ||
|
|
0d588f067f | ||
|
|
b4368fd30a | ||
|
|
b0dd007918 | ||
|
|
246ecdc11a | ||
|
|
891ea75f62 | ||
|
|
af8b401d07 | ||
|
|
3658efd98a | ||
|
|
fd044fb169 | ||
|
|
3fb0f2282c | ||
|
|
46ef98cb23 | ||
|
|
b61a4d7fa0 | ||
|
|
6417f8e63b | ||
|
|
20f752e6c1 | ||
|
|
2b37e5a29d | ||
|
|
8ec7c2108a | ||
|
|
dfc7847573 | ||
|
|
7a23eb0cc6 | ||
|
|
ab2a21ee42 | ||
|
|
249a97862f | ||
|
|
feb9ede912 | ||
|
|
03918fb091 | ||
|
|
1fee5b6bfa | ||
|
|
131d3b7f03 | ||
|
|
d794f604ac | ||
|
|
b2559731da | ||
|
|
f0c332e278 | ||
|
|
4450245d91 | ||
|
|
ca960cd80c | ||
|
|
4d721f99e2 | ||
|
|
cb2e8692a3 | ||
|
|
969ffe177f | ||
|
|
83383e2022 | ||
|
|
09aa3c8ca9 | ||
|
|
9e93795e6f | ||
|
|
3c178ea113 | ||
|
|
1f1f564f4e | ||
|
|
21cc0e3c7d | ||
|
|
40c51236db | ||
|
|
3de5f4b77a | ||
|
|
c6af472d42 | ||
|
|
1c5da9cc6a | ||
|
|
96bfb9af4e | ||
|
|
dd07526a3d | ||
|
|
fd935b9785 | ||
|
|
4e0d33c6bb | ||
|
|
5e6077dd8f | ||
|
|
41de8b9b5a | ||
|
|
5d0929efb4 | ||
|
|
5a033da83a | ||
|
|
8ff9cacae0 | ||
| f9e3bbc665 | |||
| e5f6cf8563 | |||
|
|
2ad57a8be1 | ||
|
|
4f2fbaf21b | ||
|
|
b7694d7857 | ||
|
|
b3eec37fa9 | ||
|
|
bdbf529cb6 | ||
|
|
4ff02d2bfa | ||
|
|
2ddea5468d | ||
|
|
391e11ce73 | ||
|
|
5d10cb9026 | ||
|
|
716349a839 | ||
|
|
9e0a61b7ba | ||
|
|
6ea377e9e5 | ||
|
|
8017426cfc | ||
|
|
b588479fd1 | ||
|
|
25f90e4162 | ||
|
|
b2e57396fb | ||
|
|
df786f7d34 | ||
|
|
930e120d70 | ||
|
|
0ebead27f5 | ||
|
|
397c143952 | ||
|
|
0451043a21 | ||
|
|
2cb032a5da | ||
|
|
a31ded6b02 | ||
|
|
e8e14d3b65 | ||
|
|
a6c10fe414 | ||
| b040790129 | |||
|
|
a71566f2ad | ||
|
|
cb0d0be786 | ||
|
|
2040f857a5 | ||
|
|
28a3a7a5e3 | ||
|
|
737b0b6055 | ||
|
|
bd5a01b7e4 | ||
|
|
3a74927b9d | ||
|
|
31a7b58709 | ||
|
|
acf13fb02f | ||
|
|
21773e244e | ||
|
|
345f626986 | ||
|
|
b72dcd75b7 | ||
|
|
2318a6f1e3 | ||
|
|
9b77b3f148 | ||
|
|
75ec60a1bf | ||
|
|
2c5f3a60b7 | ||
|
|
3cc79a7fd7 | ||
|
|
db7257638a | ||
|
|
59130952fa | ||
|
|
e07e6c5e19 | ||
|
|
190aae55e3 | ||
|
|
15c3747fbf | ||
|
|
d945f0ea79 | ||
|
|
99218b861a | ||
|
|
0feb51cfd0 | ||
|
|
54270c197e | ||
|
|
6940737d49 | ||
|
|
6e10a1084d | ||
|
|
ec2041b75a | ||
| d3eb10d1b3 | |||
|
|
983db5c7b8 | ||
| 95947cdefc | |||
|
|
53fe5b2e14 | ||
|
|
3905ae40ae | ||
|
|
ba17e6c72c | ||
|
|
5e95b200cf | ||
|
|
8dad0a49b1 | ||
|
|
80d3f09480 | ||
|
|
d1b4cbf28b | ||
|
|
3202d45aa8 | ||
|
|
c05de8556f | ||
|
|
81c4ca968e | ||
|
|
ba61b6718d | ||
|
|
665e19fe5b | ||
|
|
7a6274e739 | ||
|
|
f2c8794c48 | ||
|
|
593858bcd2 | ||
|
|
3605f0d1f6 | ||
|
|
f77581fc9d | ||
|
|
b74bc67a94 | ||
|
|
2f9f2169f3 | ||
|
|
0f1c040bb8 | ||
|
|
62c16785a2 | ||
|
|
1d64a6b331 | ||
|
|
ccbf7a7754 | ||
|
|
eaefc734d9 | ||
|
|
8d6c34d8b9 | ||
|
|
77c6baf46b | ||
|
|
faa4ea0152 | ||
|
|
022a3b7d8e | ||
|
|
6e8e8090fb | ||
|
|
a3403dfd92 | ||
|
|
4756745200 | ||
|
|
0d07998a9e | ||
|
|
1cbccb0463 | ||
|
|
d84ee56778 | ||
|
|
aa5aa99d7f | ||
|
|
7d22a47408 | ||
|
|
bed4422e44 | ||
|
|
6d6a4aea84 | ||
|
|
d8d7afe3e4 | ||
|
|
7b7ff8da55 | ||
|
|
fdd38dd541 | ||
|
|
5ce9d82c6d | ||
|
|
54d84cd156 | ||
|
|
5ae588e9cb | ||
|
|
8e6d5cd159 | ||
|
|
f7da73050d | ||
|
|
8a7ee09211 | ||
|
|
1ba2219710 | ||
|
|
ae83774365 | ||
|
|
47a59dc989 | ||
| 49e55088c7 | |||
|
|
19fd788c0f | ||
| 445d3ba78c | |||
|
|
3c16f8f501 | ||
|
|
41c911f867 | ||
|
|
93745dcc14 | ||
|
|
b89aebb106 | ||
|
|
a20499bf35 | ||
|
|
d06294a15a | ||
|
|
dcbc070ea5 | ||
|
|
c84fd60237 | ||
|
|
3012e026b7 | ||
|
|
d94034d5bb | ||
| dbf291070e | |||
|
|
9d2f57da76 | ||
|
|
ceda946d78 | ||
|
|
b0fe63e70f | ||
|
|
eabef7a92d | ||
|
|
9afc5b5a00 | ||
|
|
1b878d2ba0 | ||
|
|
fe33c213b1 | ||
|
|
b0b7ea0e1b | ||
|
|
3d197e877e | ||
|
|
08de15d3d8 | ||
|
|
2a1436bf7c | ||
|
|
78b0a9dddc | ||
|
|
77be7ec7fb | ||
|
|
d8309f77e1 | ||
|
|
d49a91308d | ||
|
|
96a993eeb0 | ||
|
|
760586c819 | ||
|
|
3cef741c9e | ||
|
|
d307d2e7a3 | ||
|
|
fe1129e5ff | ||
|
|
57fd3e8aab | ||
|
|
9b087e0710 | ||
|
|
568978c0dc | ||
|
|
45f477e782 | ||
|
|
79f638d01b | ||
|
|
208fb4e913 | ||
|
|
a603e3b7dc | ||
|
|
38e2340782 | ||
|
|
43f870b1bb | ||
|
|
54a59c7f0d | ||
|
|
2850a34ead | ||
|
|
9242a54fc2 | ||
|
|
b0c0510420 | ||
|
|
4dda7c8491 | ||
|
|
0fee32e788 | ||
|
|
ebf53fd79f | ||
|
|
3d639b78f2 | ||
|
|
1c1a1e6954 | ||
|
|
4e1aeacc35 | ||
|
|
4dc65fd519 | ||
|
|
bd952243b7 | ||
|
|
641d9a2519 | ||
|
|
6940d33960 | ||
| 76a4216772 | |||
| ba2291a3dd | |||
|
|
0d6f702ff5 | ||
| 32db429f29 | |||
|
|
b5d8a6d28a | ||
| d04e7322cb | |||
|
|
f5eee8b6f5 | ||
|
|
78fb70bc1a | ||
|
|
99b18617e7 | ||
|
|
6a26aed929 | ||
|
|
bdf8abfca4 | ||
|
|
fa64d4e3fd | ||
|
|
55bef7c4e7 | ||
|
|
04b51046aa | ||
|
|
6b75bd10fa | ||
|
|
1415a3c915 | ||
|
|
a3c2c85d90 | ||
|
|
38cd1593a1 | ||
|
|
815bb9df84 | ||
|
|
b5016c7b98 | ||
|
|
b968d4cba2 | ||
|
|
adda5d8c64 | ||
|
|
f15e34020e | ||
|
|
65afdeb69e | ||
|
|
1a5c591294 | ||
|
|
b5bfeab57c | ||
|
|
c309d4fbda | ||
|
|
8f2734bfaa | ||
|
|
637c13e81c | ||
|
|
3248dce03d | ||
|
|
928365dcaf | ||
|
|
e37f28096c | ||
|
|
ea1c0bf99b | ||
|
|
aff9f19d8e | ||
|
|
3324ebbb7d | ||
|
|
5633f7c71c | ||
|
|
b659cc9b15 | ||
|
|
c42591000a | ||
|
|
eaf6e084c1 | ||
|
|
ee5b69dbf3 | ||
|
|
e0a34d8d8d | ||
|
|
30e4c20669 | ||
|
|
b2f9f7158b | ||
|
|
229e017ae2 | ||
|
|
dacc7af87b | ||
|
|
e54f1fff51 | ||
|
|
cfde8b213c | ||
|
|
7d96a2c8d9 | ||
|
|
e596526762 | ||
|
|
230bfc16eb | ||
|
|
2cbc96245d | ||
|
|
952ce86ffa | ||
|
|
c62af73c16 | ||
|
|
20f665175c | ||
|
|
11c784f882 | ||
|
|
a6f4aa8d1e | ||
|
|
41e95a606d | ||
|
|
33c0fc09d0 | ||
|
|
751ae00df3 | ||
|
|
e17fb93b47 | ||
|
|
5088b9ff1b | ||
|
|
0db9a190cb | ||
|
|
27ad84fc6c | ||
|
|
447e75f97d | ||
|
|
a60c891a86 | ||
|
|
480d96771f | ||
|
|
5c620342a4 | ||
|
|
d39feb78f0 | ||
|
|
3af618d466 | ||
|
|
30f9975738 | ||
|
|
18a59b85cb | ||
|
|
2563e3ed84 | ||
|
|
c382ca3ca2 | ||
|
|
ab7ad81efb | ||
|
|
91cafe5bb9 | ||
|
|
83fb510a39 | ||
|
|
9a0ba42705 | ||
|
|
e8d625201e | ||
|
|
f1addf6746 | ||
|
|
6c50148143 | ||
|
|
8bc233a79a | ||
|
|
4abb29f8db | ||
|
|
1c976f28f0 | ||
|
|
98440bbd16 | ||
|
|
069b59e71d | ||
|
|
96b569766c | ||
|
|
939c9cbae5 | ||
|
|
be39e8fd30 | ||
|
|
279b93b61b | ||
|
|
5b796a163e | ||
|
|
305d957e11 | ||
|
|
54d1f57a54 | ||
|
|
4896284a23 | ||
|
|
5aa348be0d | ||
|
|
846cecdcab | ||
|
|
954e6b3a98 | ||
|
|
b58e0cc048 | ||
|
|
8b3c02d027 | ||
|
|
086c1c3a36 | ||
|
|
edf45774ba | ||
|
|
7f437d0587 | ||
|
|
14c818102c | ||
|
|
8ce5d43d6b | ||
|
|
b6fd8e2632 | ||
|
|
e5efdec31d | ||
|
|
de43077860 | ||
|
|
d7cb5b285e | ||
|
|
21407c0e37 | ||
|
|
484c8c3bde | ||
|
|
6499248983 | ||
|
|
f2137d310d | ||
|
|
69fd81a565 | ||
|
|
d522b03e2e | ||
|
|
e62351bbc8 | ||
|
|
2db0e10626 | ||
|
|
e331c2dcc4 | ||
|
|
eacf5ded69 | ||
|
|
bb7161b8f3 | ||
|
|
3cdff95f0f | ||
|
|
512c923095 | ||
|
|
aa522059bb | ||
|
|
15d06c6674 | ||
|
|
52eacfc8c9 | ||
|
|
78ddf742db | ||
|
|
2d2bb085cd | ||
|
|
9dc1ac5c42 | ||
|
|
3cc386acd4 | ||
|
|
887b0335b4 | ||
|
|
42931ca689 | ||
|
|
c3e44f764b | ||
|
|
8b6a27b0a7 | ||
|
|
e4ed904a8b | ||
|
|
4be1906914 | ||
|
|
eba032b3b8 | ||
|
|
666d689986 | ||
|
|
351e2b97d0 | ||
|
|
7d2aecf369 | ||
|
|
917942139f | ||
|
|
d117bdd463 | ||
|
|
68eee0f36d | ||
|
|
ecf2a16b3d | ||
|
|
eba92299cf | ||
|
|
88ff3a43a0 | ||
|
|
ba02496222 | ||
|
|
c20e151e20 | ||
|
|
2cf7f5e668 | ||
|
|
b336586de3 | ||
|
|
69557968ab | ||
|
|
9223b0dda7 | ||
|
|
146f2ba489 | ||
|
|
6403b83c23 | ||
|
|
0e111b2bb7 | ||
|
|
b2a0b351ee | ||
|
|
5be76e9d2d | ||
|
|
be124cd7a2 | ||
| 377d819038 | |||
|
|
33ad9c7703 | ||
|
|
458fd30178 | ||
| c539fc3788 | |||
| 6656676123 | |||
| 26dbfcdc72 | |||
| a256b78e5f | |||
| 9577aba145 | |||
| 2f30b995e2 | |||
| c410e9763c | |||
|
|
c308834e6d | ||
|
|
748262302c | ||
|
|
111368e9e4 | ||
|
|
70eebe58b9 | ||
|
|
67d47a6630 | ||
|
|
3dcb247ef9 | ||
|
|
15d31d6047 | ||
|
|
65ffbc358d | ||
|
|
0353d0994d | ||
|
|
160adee406 | ||
|
|
5ee5973911 | ||
|
|
331e5b0f8e | ||
|
|
4d8cf623ff | ||
|
|
9885872991 | ||
|
|
d1e2fbc750 | ||
|
|
6aa78233ff | ||
|
|
e9539cab8b | ||
|
|
386c1bbbac | ||
|
|
3bdc7afe80 | ||
|
|
1b23b83746 | ||
|
|
37fe32322e | ||
|
|
9b30e7dcfc | ||
|
|
478929d134 | ||
|
|
16e5d4b176 | ||
|
|
0b4e35308d | ||
|
|
1ec15647dc | ||
|
|
9ad4874fda | ||
|
|
4d34d3ab1d | ||
|
|
772e52358d | ||
|
|
24b59b1ecb | ||
|
|
f200be76f4 | ||
|
|
c344ef3677 | ||
|
|
388f56f183 | ||
|
|
2eab155cc9 | ||
| 3cca0d3d14 | |||
|
|
b62d21a0bf | ||
|
|
0c94647a27 | ||
|
|
2f633c72da | ||
|
|
7d6708c01b | ||
|
|
798f1cebde | ||
|
|
78aa23fab0 | ||
|
|
b6bf43d70f | ||
|
|
d9aa97c781 | ||
|
|
964a074aa4 | ||
|
|
b4aea18b63 | ||
| 4fc3a6332b | |||
| 42a1f6502f | |||
| 50c975c8bd | |||
| aed0f76dfb | |||
|
|
954e751f70 | ||
| b2f1339610 | |||
|
|
1f61e1a8ea | ||
|
|
63aacdbcaf | ||
|
|
290812bf9a | ||
|
|
be3c228e9e | ||
|
|
e585e2f5f6 | ||
|
|
ed36ce0a77 | ||
|
|
bad16943fc | ||
|
|
a21633fb7c | ||
|
|
96e220cd71 | ||
|
|
0e69d894e1 | ||
| 8160d13c86 | |||
| dcf65b87bc | |||
| a8b29ee1ce | |||
| b59d4fcdd7 | |||
| 2f730ae71e | |||
|
|
6c4845d733 | ||
|
|
91bfdc4630 | ||
|
|
ff65832012 | ||
|
|
68dd1820f0 | ||
|
|
663641a1b8 | ||
|
|
694284488d | ||
|
|
9f8693d63d | ||
|
|
21ec4d1bdd | ||
|
|
bf07c664d1 | ||
|
|
c7e56a4b13 | ||
| fc56c3f042 | |||
|
|
4ec6f1e120 | ||
|
|
1032a3b769 | ||
| cc809368f0 | |||
|
|
6051da99c8 | ||
|
|
942d7e539a | ||
|
|
85f50be9dc | ||
|
|
a52aaa11d8 | ||
|
|
5b15527d78 | ||
|
|
25fc864f66 | ||
| 0581d63942 | |||
|
|
83c247377a | ||
|
|
032dc384b7 | ||
|
|
e3cbf6af01 | ||
|
|
188ff4fb7d | ||
|
|
73e079e93a | ||
|
|
911965b48e | ||
|
|
186477f04d | ||
|
|
2b13d58077 | ||
| 4e0a1091ee | |||
|
|
a7d4103eb3 | ||
|
|
8328f5a078 | ||
|
|
7e5533e46e | ||
|
|
b041f7d0c7 | ||
|
|
3842bb6889 | ||
|
|
71d43b585a | ||
|
|
438607c222 | ||
| e79b881297 | |||
| 99f508f938 | |||
|
|
b7e1ae534a | ||
|
|
77af8f9488 | ||
|
|
dabb0cf131 | ||
|
|
55985083ec | ||
|
|
1fa234c870 | ||
|
|
7890eae03a | ||
|
|
a80225f40e | ||
|
|
467cd23457 | ||
|
|
407e9065e7 | ||
|
|
5433610a84 | ||
|
|
a42b6f4af8 | ||
|
|
fbaef0f60c | ||
|
|
342cbccff1 | ||
| 663f20cc43 | |||
|
|
73671b26ef | ||
|
|
06acec2bf2 | ||
|
|
c868f73462 | ||
|
|
93a990e229 | ||
|
|
00c587fbf2 | ||
|
|
e6fefb4049 | ||
|
|
2d5ac2664b | ||
|
|
4d4f15ce73 | ||
|
|
a30eaac775 | ||
|
|
5da4b7a4d1 | ||
|
|
4e0fcbd834 | ||
|
|
041bf6280e | ||
|
|
e37fd0da36 | ||
|
|
ca6afbb87f | ||
|
|
66f75f8722 | ||
|
|
b8ea8283eb | ||
|
|
3e0781fde6 | ||
|
|
f50658c0ba | ||
|
|
4819dfebda | ||
|
|
25fb0d6ece | ||
|
|
29fd94a0e1 | ||
|
|
851052b3d2 | ||
|
|
a43444bb3e | ||
|
|
c2a6c6872d | ||
| d5a614e5f8 | |||
| c97a172b34 | |||
|
|
31c83f5d68 | ||
|
|
1b83871cca | ||
|
|
337c2e94bc | ||
|
|
01fe9ebc9a | ||
|
|
23b2e8e439 | ||
|
|
f799498dec | ||
| 17466138dd | |||
| 63ca8e8892 | |||
| 60fdbf3d3c | |||
|
|
e8c249897d | ||
| 55d12810d9 | |||
| 639e2defa7 | |||
| b97d8ba8f1 | |||
|
|
f05b76936a | ||
|
|
0a4e1e052a | ||
|
|
28d88c282c | ||
|
|
a2be57f0aa | ||
|
|
670ecf121b | ||
|
|
79e2318256 | ||
|
|
87ba43a553 | ||
|
|
acc1d37e4f | ||
|
|
c578d039c0 | ||
|
|
162067104e | ||
|
|
ea677b6eea | ||
|
|
a82b1cfb57 | ||
| 61d0324261 | |||
|
|
4d692e0a26 | ||
|
|
9de7eada87 | ||
|
|
8f81d995a4 | ||
|
|
024f13691d | ||
|
|
4aa91cb22a | ||
|
|
9ecafe5fc8 | ||
|
|
d039e2aacd | ||
|
|
5173a6c467 | ||
|
|
1d08682393 | ||
|
|
0bb251b4d5 | ||
|
|
04baa7f6fd | ||
|
|
32212fb52d | ||
|
|
aa61a1e1b5 | ||
|
|
f0a0bacec9 | ||
|
|
5ecf9c6ba7 | ||
|
|
d233cbac22 | ||
|
|
10098f018f | ||
|
|
70de13ef6a | ||
|
|
f912fd328f | ||
|
|
6a86da2851 | ||
|
|
1a2a246bf8 | ||
|
|
248d04b666 | ||
|
|
41e96ad027 | ||
|
|
f2fd254d17 | ||
|
|
7790391b50 | ||
|
|
b827a0ab22 | ||
|
|
4142ca4fd5 | ||
|
|
fee7c2e967 | ||
|
|
091544ac3f | ||
|
|
2ce342a0cb | ||
|
|
781779328f | ||
|
|
dfced8c594 | ||
|
|
aaa4ce8372 | ||
|
|
9ddf144244 | ||
|
|
b465de55a0 | ||
|
|
cf1dc6fac8 | ||
|
|
739c63d025 | ||
|
|
8d897f824e | ||
|
|
aad129ea94 | ||
|
|
e48b29ed29 | ||
|
|
c64560bfc3 | ||
|
|
255ba86506 | ||
|
|
cd2dad9d26 | ||
|
|
c7ff6b4794 | ||
|
|
628d55df37 | ||
|
|
0059a1e7fb | ||
|
|
73ea7c0063 | ||
|
|
cb9f4befa0 | ||
|
|
decccb672a | ||
|
|
51466bf1c7 | ||
|
|
397798790e | ||
|
|
d3b8310180 | ||
|
|
7797c2b461 | ||
|
|
929e74c84b | ||
| 08e2552907 | |||
| 6617107ab8 | |||
|
|
8921959ea0 | ||
|
|
ec46cfdd65 | ||
|
|
bfb69d9a95 | ||
|
|
15f6972257 | ||
|
|
2319db3009 | ||
|
|
9cb98b5e73 | ||
|
|
2999e0e753 | ||
|
|
af716b7537 | ||
|
|
c6999c8493 | ||
|
|
d4b49a9bd7 | ||
|
|
eebc3c9806 | ||
|
|
257f1cc12d | ||
|
|
1036273a28 | ||
|
|
1b8ed93367 | ||
|
|
8c1820c879 | ||
|
|
e5a9db7e12 | ||
|
|
9d5aa7db5d | ||
|
|
40b0d9691e | ||
|
|
2e906bce79 | ||
|
|
33490dc737 | ||
|
|
f05ff4cf88 | ||
|
|
5735b49e25 | ||
|
|
1988948c30 | ||
|
|
cb33c0df68 | ||
|
|
f4ad7040ab | ||
|
|
b3b458e41c | ||
|
|
b115dc3654 | ||
|
|
8c6023e93b | ||
|
|
5d7c297f2f | ||
|
|
04f99d5ed2 | ||
|
|
bb3f21b056 | ||
|
|
cf6188d860 | ||
|
|
294bfbb35f | ||
|
|
230538bcea | ||
|
|
5e7aa72494 | ||
|
|
3159ea5665 | ||
|
|
4b307efe79 | ||
|
|
0105dab2c4 | ||
|
|
6e8e16c9a4 | ||
|
|
8d73c86946 | ||
|
|
a7b19d9bff | ||
|
|
4b7de0f690 | ||
|
|
520145bf7d | ||
|
|
e8aed880d8 | ||
|
|
babff11c70 | ||
|
|
b24c005e3e | ||
|
|
2d6c1879ce | ||
|
|
f88607c2c2 | ||
|
|
b3c13d5f98 | ||
|
|
2253d57fed | ||
|
|
0fb93f66a3 | ||
|
|
501887300d | ||
|
|
14b6247106 | ||
|
|
eda8a67718 | ||
|
|
73caf090ba | ||
|
|
afa6771903 | ||
|
|
8dd3be909b | ||
|
|
88af2469cb | ||
|
|
999d6742b4 | ||
|
|
24ac9fbf48 | ||
|
|
4350dff22a | ||
|
|
30e3d443bb | ||
|
|
f5ea7fe2c4 | ||
|
|
94d139e484 | ||
|
|
8049241f3f | ||
|
|
28d308ed4b | ||
|
|
e0862053d0 | ||
|
|
e0e76d3a56 | ||
|
|
242e44315b | ||
|
|
6b53ed13e2 | ||
|
|
fafccbcecc | ||
|
|
49698bfda9 | ||
|
|
4154cc2230 | ||
|
|
ce8a96b958 | ||
|
|
f373a3abc7 | ||
|
|
371ee0a780 | ||
|
|
327cf5730b | ||
|
|
5e1a373730 | ||
|
|
717a7dc95f | ||
|
|
aa2417af12 | ||
|
|
f2533f1ed5 | ||
|
|
7d41699207 | ||
|
|
f9ccfef8e0 | ||
|
|
468a2ba251 | ||
|
|
08dbdcf4ae | ||
|
|
50d73f7bc8 | ||
|
|
d7ed9ff592 | ||
|
|
05c6fd4528 | ||
|
|
a99f424901 | ||
|
|
648354500b | ||
|
|
28c0480b8b | ||
|
|
f29ed6e29b | ||
|
|
903b4d79ac | ||
|
|
026c189901 | ||
|
|
f2b30ee349 | ||
|
|
5c02a9325a | ||
|
|
7645c8642f | ||
|
|
c5ebbe319f | ||
|
|
6f36eb1690 | ||
|
|
18f9e006a3 | ||
|
|
a7e334ebe9 | ||
|
|
a5e15d9d84 | ||
|
|
b3173517b4 | ||
|
|
06d2963163 | ||
|
|
df8282015d | ||
|
|
81b0aab850 | ||
|
|
a8705c5b44 | ||
|
|
efe739f9fc | ||
|
|
2e83a0508a | ||
|
|
d66adb2a1f | ||
|
|
e5cecdee03 | ||
|
|
9351af16b7 | ||
|
|
ad8831a75a | ||
|
|
c63018ff08 | ||
|
|
131bce3898 | ||
|
|
fff9316030 | ||
|
|
2b3148dc3c | ||
|
|
13a4202f08 | ||
|
|
f0df143aaf | ||
|
|
70a94ab2c6 | ||
|
|
e6d18d1a72 | ||
|
|
46cd57fb36 | ||
|
|
4ae389846d | ||
|
|
37d04e9e89 | ||
|
|
39d141ca67 | ||
|
|
befcc169e0 | ||
|
|
6bbb9c1b81 | ||
|
|
bb77afc4d2 | ||
|
|
b43697fd68 | ||
|
|
86eebbead6 | ||
|
|
7463b8badc | ||
|
|
37e857f093 | ||
|
|
2db240b32e | ||
|
|
3c5f6031e1 | ||
|
|
b25fee0d86 | ||
|
|
a6ee58ff28 | ||
|
|
fdb083f2a3 | ||
|
|
ee32a35a42 | ||
|
|
d17b41da38 | ||
|
|
870184b2a5 | ||
|
|
e787691efc | ||
|
|
859716e6c7 | ||
|
|
aab91b3842 | ||
|
|
f9dd0a6b03 | ||
|
|
f55cfae6e4 | ||
|
|
671c5b37f3 | ||
|
|
46c4ec7ab9 | ||
|
|
1856bd5022 | ||
|
|
21cfc498e8 | ||
|
|
ec7007071e | ||
|
|
e33509789c | ||
|
|
9fddcef271 | ||
|
|
d2acb641c5 | ||
|
|
82a6b634a9 | ||
|
|
f466b0cf7b | ||
|
|
b3f6fc852b | ||
|
|
976d892061 | ||
|
|
2a0a9e79da | ||
|
|
8d24093aec | ||
|
|
c6b23d353a | ||
|
|
b6a21b9c12 | ||
|
|
14fbae3c86 | ||
|
|
1dd88d117d | ||
|
|
f6d70b4f7d | ||
|
|
368bf29e63 | ||
|
|
c40c0c54bd | ||
|
|
0c850d47d7 | ||
|
|
4879df040f | ||
|
|
7cf013e52b | ||
|
|
4e0a1a59e0 | ||
|
|
6953c954a8 | ||
|
|
e3664cc973 | ||
|
|
417d938677 | ||
|
|
2042abb045 | ||
|
|
0f5efa9852 | ||
|
|
76968ded02 | ||
|
|
ce8e7be9d7 | ||
|
|
1290d6b478 | ||
|
|
a5c38658a8 | ||
|
|
5797beb0ca | ||
|
|
9178bc083e | ||
|
|
878840fc06 | ||
|
|
716f0587c3 | ||
|
|
03eed5131a | ||
|
|
3946c88ed7 | ||
|
|
59182f20f3 | ||
|
|
a20b128ee8 | ||
|
|
1edd9ab371 | ||
|
|
da8defc517 | ||
|
|
45272fb34f | ||
|
|
a8dc44b41e | ||
|
|
1954a5a90a | ||
|
|
228530a454 | ||
|
|
4216cdd1e5 | ||
|
|
6afb019f9d | ||
|
|
8410309e38 | ||
|
|
ceb6faa5af | ||
|
|
9de4777cdb | ||
|
|
83df967247 | ||
|
|
39e2356feb | ||
|
|
24484015b3 | ||
|
|
30e7b220cd | ||
|
|
576092fc96 | ||
|
|
2114800b84 | ||
|
|
d9492eeee6 | ||
|
|
2442b59a5f | ||
|
|
1c01e27a65 | ||
|
|
0a9b1907d7 | ||
|
|
0a9e14c460 | ||
|
|
33797ce9de | ||
|
|
97ff1402f8 | ||
|
|
5fa6660afd | ||
|
|
c05470c045 | ||
|
|
e24e648e07 | ||
|
|
a4d51304b9 | ||
|
|
4210214735 | ||
|
|
403dea665d | ||
|
|
5704300d75 | ||
|
|
deb238d423 | ||
|
|
b678041e68 | ||
|
|
df5fe085a8 | ||
|
|
46ee2c6dd0 | ||
|
|
eccbe59a1a | ||
|
|
3e2869b54a | ||
|
|
96b8a7626c | ||
|
|
7593d6e564 | ||
|
|
e00aacfe3e | ||
|
|
860c2bc8fb | ||
|
|
710f1b9f90 | ||
|
|
4e87ca790e | ||
|
|
5974f3c9f2 | ||
|
|
62f1b7eb88 | ||
|
|
1e9574e740 | ||
|
|
b6207ba447 | ||
|
|
7fe2611aed | ||
|
|
860bd8539f | ||
|
|
ceade85334 | ||
|
|
5dc81f60c0 | ||
|
|
40b08622ec | ||
|
|
a12d5dbb78 |
8
.c8rc.json
Normal file
8
.c8rc.json
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"src": "./src",
|
||||||
|
"exclude": ["**/bot/lib", "__mocks__", "**/mock"],
|
||||||
|
"reporter": ["html"],
|
||||||
|
"report-dir": "./coverage",
|
||||||
|
"check-coverage": true,
|
||||||
|
"lines": 90
|
||||||
|
}
|
||||||
13
.env.example
13
.env.example
@@ -1,13 +0,0 @@
|
|||||||
######DATABASE: none, mysql, dialogflow
|
|
||||||
|
|
||||||
DEFAULT_MESSAGE=true
|
|
||||||
SAVE_MEDIA=true
|
|
||||||
PORT=3000
|
|
||||||
DATABASE=none
|
|
||||||
LANGUAGE=es
|
|
||||||
SQL_HOST=
|
|
||||||
SQL_USER=
|
|
||||||
SQL_PASS=
|
|
||||||
SQL_DATABASE=
|
|
||||||
KEEP_DIALOG_FLOW=false
|
|
||||||
MULTI_DEVICE=true
|
|
||||||
2
.eslintignore
Normal file
2
.eslintignore
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
packages/docs/*
|
||||||
|
packages/portal/*
|
||||||
18
.eslintrc.js
Normal file
18
.eslintrc.js
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
module.exports = {
|
||||||
|
env: {
|
||||||
|
browser: true,
|
||||||
|
commonjs: true,
|
||||||
|
es2021: true,
|
||||||
|
node: true,
|
||||||
|
},
|
||||||
|
extends: 'eslint:recommended',
|
||||||
|
overrides: [],
|
||||||
|
parserOptions: {
|
||||||
|
ecmaVersion: 'latest',
|
||||||
|
},
|
||||||
|
rules: {
|
||||||
|
'no-unsafe-negation': 'off',
|
||||||
|
'no-prototype-builtins': 'off',
|
||||||
|
'no-useless-escape': 'off',
|
||||||
|
},
|
||||||
|
}
|
||||||
4
.github/FUNDING.yml
vendored
4
.github/FUNDING.yml
vendored
@@ -1,2 +1,4 @@
|
|||||||
|
# These are supported funding model platforms
|
||||||
|
open_collective: bot-whatsapp
|
||||||
patreon: leifermendez
|
patreon: leifermendez
|
||||||
custom: "https://www.buymeacoffee.com/leifermendez"
|
custom: https://www.buymeacoffee.com/leifermendez
|
||||||
|
|||||||
58
.github/ISSUE_TEMPLATE/bug.yml
vendored
Normal file
58
.github/ISSUE_TEMPLATE/bug.yml
vendored
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
name: 🐛 Reporte Bug
|
||||||
|
description: Algo no va bien?. Hazlo saber
|
||||||
|
labels: [bug, triage]
|
||||||
|
title: '[🐛]'
|
||||||
|
body:
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
Gracias por tomarte el tiempo de reportar este problema
|
||||||
|
|
||||||
|
- type: dropdown
|
||||||
|
id: version
|
||||||
|
attributes:
|
||||||
|
label: ¿Que versión estas usando?
|
||||||
|
description: '__INFO:__ Recuerda que puedes consultar dudas directamente en [discord](https://link.codigoencasa.com/DISCORD)'
|
||||||
|
options:
|
||||||
|
- v2
|
||||||
|
- v1
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: dropdown
|
||||||
|
id: component
|
||||||
|
attributes:
|
||||||
|
label: ¿Sobre que afecta?
|
||||||
|
options:
|
||||||
|
- Flujo de palabras (Flow)
|
||||||
|
- DialogFlow
|
||||||
|
- Base de datos
|
||||||
|
- Otro
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: description
|
||||||
|
attributes:
|
||||||
|
description: 'Trata de ser lo más claro posible, de esa manera podemos entender el contexto de tu problema y darte una mejor solución'
|
||||||
|
label: Describe tu problema
|
||||||
|
placeholder: Yo tengo un problema....
|
||||||
|
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: input
|
||||||
|
id: reproduction
|
||||||
|
attributes:
|
||||||
|
label: Reproducir error
|
||||||
|
description: __(Recomendación)__ trata de grabar un video puedes usar algunas de las siguientes herramientas [https://www.vidyard.com/](https://www.vidyard.com/) [https://www.loom.com/](https://www.loom.com/) y en lo posbile apoyate en [https://stackblitz.com/](https://stackblitz.com/) para compartir el código de ser necesario
|
||||||
|
placeholder: URL video o stackblitz
|
||||||
|
validations:
|
||||||
|
required: false
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: additional_information
|
||||||
|
attributes:
|
||||||
|
label: Información Adicional
|
||||||
|
validations:
|
||||||
|
required: false
|
||||||
4
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
4
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
contact_links:
|
||||||
|
- name: 🤔 Core Team
|
||||||
|
url: https://link.codigoencasa.com/DISCORD
|
||||||
|
about: Si quieres formar parte del CoreTeam, patrocinar el proyecto o propuesta profesionales
|
||||||
79
.github/ISSUE_TEMPLATE/test-case.yml
vendored
Normal file
79
.github/ISSUE_TEMPLATE/test-case.yml
vendored
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
name: 🐬 Caso de uso
|
||||||
|
description: Reporta tu caso de uso y cuales fueron tus resultados
|
||||||
|
labels: [usecase]
|
||||||
|
title: '[🐬]'
|
||||||
|
body:
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
Gracias por tomarte el tiempo de detallar este caso de uso, sera de gran utilidad para mantener un software de calidad puedes comenzar
|
||||||
|
⚡ `npm create bot-whatsapp@dev`
|
||||||
|
|
||||||
|
- type: dropdown
|
||||||
|
id: version
|
||||||
|
attributes:
|
||||||
|
label: ¿Cual proveedor usaste?
|
||||||
|
description: 'Actualmente tenemos varios proveedores que sirven como punto de entrada y salida con Whatsapp'
|
||||||
|
options:
|
||||||
|
- whatsapp-web.js
|
||||||
|
- venom
|
||||||
|
- bailey
|
||||||
|
- twilio
|
||||||
|
- meta
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: dropdown
|
||||||
|
id: component
|
||||||
|
attributes:
|
||||||
|
label: ¿Cual base de datos usaste?
|
||||||
|
options:
|
||||||
|
- memory
|
||||||
|
- mongo
|
||||||
|
- mysql
|
||||||
|
- json
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: dropdown
|
||||||
|
id: result
|
||||||
|
attributes:
|
||||||
|
label: Conclusion de la prueba
|
||||||
|
options:
|
||||||
|
- muy buena
|
||||||
|
- buena
|
||||||
|
- tiene errores
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: description
|
||||||
|
attributes:
|
||||||
|
description: 'Trata de ser lo más claro posible, de esa manera podemos entender el contexto del caso de uso'
|
||||||
|
label: Describe tu caso
|
||||||
|
placeholder: Yo tengo un caso....
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: logs
|
||||||
|
attributes:
|
||||||
|
label: ¿Logs Importantes?
|
||||||
|
description: Si tienes algunos logs importantes a tener en cuenta o que muetren algun error en concreto.
|
||||||
|
render: shell
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: additional_information
|
||||||
|
attributes:
|
||||||
|
label: Información Adicional
|
||||||
|
validations:
|
||||||
|
required: false
|
||||||
|
|
||||||
|
- type: input
|
||||||
|
id: usernames
|
||||||
|
attributes:
|
||||||
|
label: ¿Quieres que te mencionemos?
|
||||||
|
description: Siempre buscamos fomentar la comunidad por lo cual si quieres que te mencionemos publicamente en nuestras redes sociales puedes dejar tu username
|
||||||
|
placeholder: twitter o github o instagram o alguna url
|
||||||
|
validations:
|
||||||
|
required: false
|
||||||
17
.github/PULL_REQUEST_TEMPLATE.md
vendored
Normal file
17
.github/PULL_REQUEST_TEMPLATE.md
vendored
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
# Que tipo de Pull Request es?
|
||||||
|
|
||||||
|
- [ ] Mejoras
|
||||||
|
- [ ] Bug
|
||||||
|
- [ ] Docs / tests
|
||||||
|
|
||||||
|
# Descripción
|
||||||
|
|
||||||
|
Por favor agrega una descripción de tu aporte para tener más contexto y poder avanzar más rápido. Si es de ayuda puedes usar plataformar como [https://www.loom.com/](https://www.loom.com/) para grabar un video.
|
||||||
|
|
||||||
|
|
||||||
|
> Forma parte de este proyecto.
|
||||||
|
|
||||||
|
- [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)
|
||||||
83
.github/workflows/ci.yml
vendored
Normal file
83
.github/workflows/ci.yml
vendored
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
name: Build and Test
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- dev
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
############ BUILD PACKAGE ############
|
||||||
|
build-package:
|
||||||
|
name: Build Package
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs:
|
||||||
|
- test-unit
|
||||||
|
- test-e2e
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 16.x
|
||||||
|
cache: 'yarn'
|
||||||
|
registry-url: https://registry.npmjs.org/
|
||||||
|
|
||||||
|
- run: corepack enable
|
||||||
|
|
||||||
|
- name: Install NPM Dependencies
|
||||||
|
run: yarn install --immutable --network-timeout 300000
|
||||||
|
|
||||||
|
- name: Build Package
|
||||||
|
run: yarn build:full
|
||||||
|
|
||||||
|
- name: Build Eslint rules
|
||||||
|
run: yarn lint:fix
|
||||||
|
|
||||||
|
############ UNIT TEST ############
|
||||||
|
test-unit:
|
||||||
|
name: Unit Tests
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 16.x
|
||||||
|
cache: 'yarn'
|
||||||
|
registry-url: https://registry.npmjs.org/
|
||||||
|
|
||||||
|
- run: corepack enable
|
||||||
|
|
||||||
|
- name: Install NPM Dependencies
|
||||||
|
run: yarn install --immutable --network-timeout 300000
|
||||||
|
|
||||||
|
- name: Unit Tests
|
||||||
|
run: yarn test
|
||||||
|
############ UNIT TEST ############
|
||||||
|
test-e2e:
|
||||||
|
name: e2e Tests
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs:
|
||||||
|
- test-unit
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 16.x
|
||||||
|
cache: 'yarn'
|
||||||
|
registry-url: https://registry.npmjs.org/
|
||||||
|
|
||||||
|
- run: corepack enable
|
||||||
|
|
||||||
|
- name: Install NPM Dependencies
|
||||||
|
run: yarn install --immutable --network-timeout 300000
|
||||||
|
|
||||||
|
- name: e2e Tests
|
||||||
|
run: yarn test.e2e
|
||||||
76
.github/workflows/codeql.yml
vendored
Normal file
76
.github/workflows/codeql.yml
vendored
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
# For most projects, this workflow file will not need changing; you simply need
|
||||||
|
# to commit it to your repository.
|
||||||
|
#
|
||||||
|
# You may wish to alter this file to override the set of languages analyzed,
|
||||||
|
# or to provide custom queries or build logic.
|
||||||
|
#
|
||||||
|
# ******** NOTE ********
|
||||||
|
# We have attempted to detect the languages in your repository. Please check
|
||||||
|
# the `language` matrix defined below to confirm you have the correct set of
|
||||||
|
# supported CodeQL languages.
|
||||||
|
#
|
||||||
|
name: 'CodeQL'
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [release/next]
|
||||||
|
pull_request:
|
||||||
|
# The branches below must be a subset of the branches above
|
||||||
|
branches: ['main']
|
||||||
|
schedule:
|
||||||
|
- cron: '21 16 * * 5'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
analyze:
|
||||||
|
if: ${{ !github.event.act }}
|
||||||
|
name: Analyze
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
actions: read
|
||||||
|
contents: read
|
||||||
|
security-events: write
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
language: ['javascript']
|
||||||
|
# CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ]
|
||||||
|
# Use only 'java' to analyze code written in Java, Kotlin or both
|
||||||
|
# Use only 'javascript' to analyze code written in JavaScript, TypeScript or both
|
||||||
|
# Learn more about CodeQL language support at https://aka.ms/codeql-docs/language-support
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
# Initializes the CodeQL tools for scanning.
|
||||||
|
- name: Initialize CodeQL
|
||||||
|
uses: github/codeql-action/init@v2
|
||||||
|
with:
|
||||||
|
languages: ${{ matrix.language }}
|
||||||
|
# If you wish to specify custom queries, you can do so here or in a config file.
|
||||||
|
# By default, queries listed here will override any specified in a config file.
|
||||||
|
# Prefix the list here with "+" to use these queries and those in the config file.
|
||||||
|
|
||||||
|
# Details on CodeQL's query packs refer to : https://docs.github.com/en/code-security/code-scanning/automatically-scanning-your-code-for-vulnerabilities-and-errors/configuring-code-scanning#using-queries-in-ql-packs
|
||||||
|
# queries: security-extended,security-and-quality
|
||||||
|
|
||||||
|
# Autobuild attempts to build any compiled languages (C/C++, C#, Go, or Java).
|
||||||
|
# If this step fails, then you should remove it and run the build manually (see below)
|
||||||
|
- name: Autobuild
|
||||||
|
uses: github/codeql-action/autobuild@v2
|
||||||
|
|
||||||
|
# ℹ️ Command-line programs to run using the OS shell.
|
||||||
|
# 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun
|
||||||
|
|
||||||
|
# If the Autobuild fails above, remove it and uncomment the following three lines.
|
||||||
|
# modify them (or add more) to build your code if your project, please refer to the EXAMPLE below for guidance.
|
||||||
|
|
||||||
|
# - run: |
|
||||||
|
# echo "Run, Build Application using script"
|
||||||
|
# ./location_of_script_within_repo/buildscript.sh
|
||||||
|
|
||||||
|
- name: Perform CodeQL Analysis
|
||||||
|
uses: github/codeql-action/analyze@v2
|
||||||
|
with:
|
||||||
|
category: '/language:${{matrix.language}}'
|
||||||
47
.github/workflows/netlify-dev.yml
vendored
Normal file
47
.github/workflows/netlify-dev.yml
vendored
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
name: 📄 Desplegando documentacion
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- 'feat/docs-**'
|
||||||
|
- 'fix/docs-**'
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- 'feat/docs-**'
|
||||||
|
- 'fix/docs-**'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
############ DOCUMENTATION BUILD ############
|
||||||
|
build-documentation:
|
||||||
|
if: ${{ !github.event.act }}
|
||||||
|
name: Build Package
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 16.x
|
||||||
|
cache: 'yarn'
|
||||||
|
registry-url: https://registry.npmjs.org/
|
||||||
|
|
||||||
|
- run: corepack enable
|
||||||
|
|
||||||
|
- name: Install NPM Dependencies
|
||||||
|
run: yarn install --immutable --network-timeout 300000
|
||||||
|
|
||||||
|
- name: Add netlify
|
||||||
|
run: yarn add netlify-cli -D
|
||||||
|
|
||||||
|
- name: Create .env build file
|
||||||
|
run: |
|
||||||
|
touch packages/docs/.env
|
||||||
|
echo VITE_GITHUB_TOKEN=${{ secrets.COLLABORATORS_TOKEN }} >> packages/docs/.env
|
||||||
|
|
||||||
|
- name: Build and Deploy
|
||||||
|
run: |
|
||||||
|
cd packages/docs
|
||||||
|
netlify deploy --build --site ${{ secrets.NETLIFY_SITE_ID }} --auth ${{ secrets.NETLIFY_AUTH_TOKEN }}
|
||||||
41
.github/workflows/netlify.yml
vendored
Normal file
41
.github/workflows/netlify.yml
vendored
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
name: 📄 (PROD) Desplegando documentacion
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- release/next
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
############ DOCUMENTATION BUILD ############
|
||||||
|
build-documentation-prod:
|
||||||
|
name: Build Package
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 16.x
|
||||||
|
cache: 'yarn'
|
||||||
|
registry-url: https://registry.npmjs.org/
|
||||||
|
|
||||||
|
- run: corepack enable
|
||||||
|
|
||||||
|
- name: Install NPM Dependencies
|
||||||
|
run: yarn install --immutable --network-timeout 300000
|
||||||
|
|
||||||
|
- name: Add netlify
|
||||||
|
run: yarn add netlify-cli -D
|
||||||
|
|
||||||
|
- name: Create .env build file
|
||||||
|
run: |
|
||||||
|
touch packages/docs/.env
|
||||||
|
echo VITE_GITHUB_TOKEN=${{ secrets.COLLABORATORS_TOKEN }} >> packages/docs/.env
|
||||||
|
|
||||||
|
- name: Build and Deploy
|
||||||
|
run: |
|
||||||
|
cd packages/docs
|
||||||
|
netlify deploy --prod --build --site ${{ secrets.NETLIFY_SITE_ID }} --auth ${{ secrets.NETLIFY_AUTH_TOKEN }}
|
||||||
64
.github/workflows/releases-dev.yml
vendored
Normal file
64
.github/workflows/releases-dev.yml
vendored
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
name: 🚀 (DEV) Liberando versiones
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- release/next
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
############ RELEASE ############
|
||||||
|
release:
|
||||||
|
name: Release
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
outputs:
|
||||||
|
commit: ${{ steps.vars.outputs.commit }}
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
ref: ${{github.event.after}}
|
||||||
|
persist-credentials: false
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 16.x
|
||||||
|
cache: 'yarn'
|
||||||
|
registry-url: https://registry.npmjs.org/
|
||||||
|
|
||||||
|
- run: corepack enable
|
||||||
|
|
||||||
|
- name: Install NPM Dependencies
|
||||||
|
run: yarn install --immutable --network-timeout 300000
|
||||||
|
|
||||||
|
- name: Build Package
|
||||||
|
run: yarn build:full
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/bot
|
||||||
|
run: yarn node ./scripts/release.js --name=bot --version= --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/cli
|
||||||
|
run: yarn node ./scripts/release.js --name=cli --version= --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/create-bot-whatsapp
|
||||||
|
run: yarn node ./scripts/release.js --name=create-bot-whatsapp --version= --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/database
|
||||||
|
run: yarn node ./scripts/release.js --name=database --version= --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/provider
|
||||||
|
run: yarn node ./scripts/release.js --name=provider --version= --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/contexts
|
||||||
|
run: yarn node ./scripts/release.js --name=contexts --version= --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/portal
|
||||||
|
run: yarn node ./scripts/release.js --name=portal --version= --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Commit & Push changes
|
||||||
|
uses: actions-js/push@master
|
||||||
|
with:
|
||||||
|
branch: release/next
|
||||||
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
force: true
|
||||||
75
.github/workflows/releases.yml
vendored
Normal file
75
.github/workflows/releases.yml
vendored
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
name: 🚀⚡ Liberando versiones
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- release/production
|
||||||
|
jobs:
|
||||||
|
############ RELEASE ############
|
||||||
|
release-prod:
|
||||||
|
if: ${{ !github.event.act }}
|
||||||
|
name: Release
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Set output
|
||||||
|
id: vars
|
||||||
|
run: echo "tag=${GITHUB_REF#refs/*/}" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 16.x
|
||||||
|
cache: 'yarn'
|
||||||
|
registry-url: https://registry.npmjs.org/
|
||||||
|
|
||||||
|
- run: corepack enable
|
||||||
|
|
||||||
|
- name: Set User
|
||||||
|
run: git config --global user.email "leifer.contacto@gmail.com" && git config --global user.name "Leifer Mendez"
|
||||||
|
|
||||||
|
- name: Install NPM Dependencies
|
||||||
|
run: yarn install --immutable --network-timeout 300000
|
||||||
|
|
||||||
|
- name: Set CHANGELOG
|
||||||
|
run: yarn release
|
||||||
|
|
||||||
|
- name: get-npm-version
|
||||||
|
id: package-version
|
||||||
|
uses: martinbeentjes/npm-get-version-action@main
|
||||||
|
|
||||||
|
- name: Build Package
|
||||||
|
run: yarn build:full
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/bot
|
||||||
|
run: yarn node ./scripts/release.js --name=bot --version="${{ steps.package-version.outputs.current-version}}" --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/cli
|
||||||
|
run: yarn node ./scripts/release.js --name=cli --version="${{ steps.package-version.outputs.current-version}}" --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/create-bot-whatsapp
|
||||||
|
run: yarn node ./scripts/release.js --name=create-bot-whatsapp --version="${{ steps.package-version.outputs.current-version}}" --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/database
|
||||||
|
run: yarn node ./scripts/release.js --name=database --version="${{ steps.package-version.outputs.current-version}}" --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/provider
|
||||||
|
run: yarn node ./scripts/release.js --name=provider --version="${{ steps.package-version.outputs.current-version}}" --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/contexts
|
||||||
|
run: yarn node ./scripts/release.js --name=contexts --version="${{ steps.package-version.outputs.current-version}}" --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release @bot-whatsapp/portal
|
||||||
|
run: yarn node ./scripts/release.js --name=portal --version="${{ steps.package-version.outputs.current-version}}" --token="${{ secrets.NPM_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Release Github
|
||||||
|
run: yarn node ./scripts/github.js --version="${{ steps.package-version.outputs.current-version}}" --token="${{ secrets.OCTO_TOKEN }}"
|
||||||
|
|
||||||
|
- name: Commit & Push changes
|
||||||
|
uses: actions-js/push@master
|
||||||
|
with:
|
||||||
|
branch: release/production
|
||||||
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
force: true
|
||||||
27
.github/workflows/stale.yml
vendored
Normal file
27
.github/workflows/stale.yml
vendored
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
# This workflow warns and then closes issues and PRs that have had no activity for a specified amount of time.
|
||||||
|
#
|
||||||
|
# You can adjust the behavior by modifying this file.
|
||||||
|
# For more information, see:
|
||||||
|
# https://github.com/actions/stale
|
||||||
|
name: Revisar ISSUES abandonadas
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '55 22 * * *'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
stale:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
issues: write
|
||||||
|
pull-requests: write
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/stale@v5
|
||||||
|
with:
|
||||||
|
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
stale-issue-message: '¿Alguna novedad sobre esta ISSUE?'
|
||||||
|
stale-pr-message: '¿Alguna novedad sobre esta PULL REQUEST?'
|
||||||
|
stale-issue-label: 'no-issue-activity'
|
||||||
|
stale-pr-label: 'no-pr-activity'
|
||||||
|
exempt-issue-assignees: 'leifermendez'
|
||||||
37
.gitignore
vendored
37
.gitignore
vendored
@@ -1,5 +1,10 @@
|
|||||||
/node_modules
|
/node_modules
|
||||||
/node_modules/*
|
/packages/repl
|
||||||
|
/packages/*/starters
|
||||||
|
/packages/*/node_modules
|
||||||
|
/packages/*/dist
|
||||||
|
/packages/*/docs/dist
|
||||||
|
/packages/provider/src/venom/tokens
|
||||||
session.json
|
session.json
|
||||||
chats/*
|
chats/*
|
||||||
!chats/.gitkeep
|
!chats/.gitkeep
|
||||||
@@ -10,3 +15,33 @@ mediaSend/*
|
|||||||
!mediaSend/nota-de-voz.mp3
|
!mediaSend/nota-de-voz.mp3
|
||||||
.env
|
.env
|
||||||
.wwebjs_auth
|
.wwebjs_auth
|
||||||
|
/session
|
||||||
|
/session/*
|
||||||
|
/tokens
|
||||||
|
/tokens/*
|
||||||
|
packages/cli/config.json
|
||||||
|
config.json
|
||||||
|
.yarnrc.yml
|
||||||
|
coverage/
|
||||||
|
*.lcov
|
||||||
|
log
|
||||||
|
log/*
|
||||||
|
*.log
|
||||||
|
*.tgz
|
||||||
|
lib
|
||||||
|
tmp/
|
||||||
|
.yarn/*
|
||||||
|
!.yarn/releases
|
||||||
|
!.yarn/plugins/@yarnpkg/plugin-postinstall.cjs
|
||||||
|
.fleet/
|
||||||
|
example-app*/
|
||||||
|
base-*/
|
||||||
|
test-*.json
|
||||||
|
!starters/apps/base-*/
|
||||||
|
qr.svg
|
||||||
|
package-lock.json
|
||||||
|
yarn-error.log
|
||||||
|
.npmrc
|
||||||
|
# Local Netlify folder
|
||||||
|
.netlify
|
||||||
|
.secrets
|
||||||
4
.husky/commit-msg
Executable file
4
.husky/commit-msg
Executable file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/usr/bin/env sh
|
||||||
|
. "$(dirname -- "$0")/_/husky.sh"
|
||||||
|
|
||||||
|
npm run lint:fix && npx --no -- commitlint --edit
|
||||||
4
.husky/pre-commit
Executable file
4
.husky/pre-commit
Executable file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/usr/bin/env sh
|
||||||
|
. "$(dirname -- "$0")/_/husky.sh"
|
||||||
|
|
||||||
|
yarn run fmt.staged
|
||||||
4
.husky/pre-push
Executable file
4
.husky/pre-push
Executable file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/usr/bin/env sh
|
||||||
|
. "$(dirname -- "$0")/_/husky.sh"
|
||||||
|
|
||||||
|
npm run test
|
||||||
9
.prettierignore
Normal file
9
.prettierignore
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
packages/**/lib
|
||||||
|
packages/docs/
|
||||||
|
**/.git
|
||||||
|
**/.svn
|
||||||
|
**/.hg
|
||||||
|
**/node_modules
|
||||||
|
*.mjs
|
||||||
|
*.cjs
|
||||||
|
*.md
|
||||||
@@ -1 +1,7 @@
|
|||||||
|
{
|
||||||
|
"trailingComma": "es5",
|
||||||
|
"tabWidth": 4,
|
||||||
|
"semi": false,
|
||||||
|
"singleQuote": true,
|
||||||
|
"printWidth": 120
|
||||||
|
}
|
||||||
|
|||||||
24
.vscode/bot.code-snippets
vendored
Normal file
24
.vscode/bot.code-snippets
vendored
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
{
|
||||||
|
"Flow Bot (simple)": {
|
||||||
|
"scope": "javascript",
|
||||||
|
"prefix": "bot:flow",
|
||||||
|
"description": "Crear un flujo simple",
|
||||||
|
"body": [
|
||||||
|
"export const flow${1} = addKeyword(['hola', 'buenas'])",
|
||||||
|
" .addAnswer('Hola! 🚀 Bienvenido a este CHATBOT')",
|
||||||
|
" .addAnswer('¿Como puedo ayudarte?')"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Flow Bot (completo)": {
|
||||||
|
"scope": "javascript",
|
||||||
|
"prefix": "bot:flow completo",
|
||||||
|
"description": "Crear un flujo completo",
|
||||||
|
"body": [
|
||||||
|
"export const flow${1} = addKeyword(['categorias'])",
|
||||||
|
" .addAnswer('⚡ Tenemos las siguientes categorias')",
|
||||||
|
" .addAnswer(['🚀 Computadoras', '🚀 Celulares', '🚀 Otros'], {",
|
||||||
|
" delay: 1500, //Milisegundo 1500 = 1.5segundos",
|
||||||
|
"})"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
3
.vscode/extensions.json
vendored
Normal file
3
.vscode/extensions.json
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"recommendations": ["xyc.vscode-mdx-preview", "vivaxy.vscode-conventional-commits", "mhutchie.git-graph"]
|
||||||
|
}
|
||||||
15
.vscode/launch.json
vendored
Normal file
15
.vscode/launch.json
vendored
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
// Use IntelliSense para saber los atributos posibles.
|
||||||
|
// Mantenga el puntero para ver las descripciones de los existentes atributos.
|
||||||
|
// Para más información, visite: https://go.microsoft.com/fwlink/?linkid=830387
|
||||||
|
"version": "0.2.0",
|
||||||
|
"configurations": [
|
||||||
|
{
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"name": "Iniciar el programa",
|
||||||
|
"skipFiles": ["<node_internals>/**"],
|
||||||
|
"program": "${workspaceFolder}\\example-app\\app.js"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
14
.vscode/settings.json
vendored
Normal file
14
.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
"conventionalCommits.scopes": [
|
||||||
|
"hook",
|
||||||
|
"contributing",
|
||||||
|
"cli",
|
||||||
|
"bot",
|
||||||
|
"provider",
|
||||||
|
"adapter",
|
||||||
|
"ci",
|
||||||
|
"starters",
|
||||||
|
"conflict",
|
||||||
|
"contexts"
|
||||||
|
]
|
||||||
|
}
|
||||||
8
.yarn/plugins/@yarnpkg/plugin-postinstall.cjs
vendored
Normal file
8
.yarn/plugins/@yarnpkg/plugin-postinstall.cjs
vendored
Normal 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
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
10
.yarnrc.yml
Normal 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
|
||||||
588
CHANGELOG.md
588
CHANGELOG.md
@@ -1,4 +1,592 @@
|
|||||||
|
# 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.1.21](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.20...v0.1.21) (2023-02-12)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* **provider:** :bug: add_location ([a46a9ef](https://github.com/leifermendez/bot-whatsapp/commit/a46a9efd8dbd921c773395d331154dc9a8aae783))
|
||||||
|
* **provider:** :rocket: feat Instance provider ([2278149](https://github.com/leifermendez/bot-whatsapp/commit/227814929561cedc11a1f69c8029515a7f47c9ff))
|
||||||
|
* **provider:** :rocket: provider raw ([2d44a00](https://github.com/leifermendez/bot-whatsapp/commit/2d44a002ff226fb0eb7362ad49936f1e00b84242))
|
||||||
|
* **provider:** :zap: add location provider ([c7de860](https://github.com/leifermendez/bot-whatsapp/commit/c7de860803fb362f5afe06cc38ad71b2c316d524))
|
||||||
|
* **provider:** :zap: add location provider ([c0ece6f](https://github.com/leifermendez/bot-whatsapp/commit/c0ece6feb2b0325476880a604c32de341eb60544))
|
||||||
|
* **provider:** :zap: support location <20> ([a147677](https://github.com/leifermendez/bot-whatsapp/commit/a147677a26b36bba429c3dd3c2c81a44a7a4d2b6))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* arreglando dir y varios mensajes en dialog flow essential ([01c7db8](https://github.com/leifermendez/bot-whatsapp/commit/01c7db8fe7dda2482eb0aa1fd7b3469b6ae8eae1))
|
||||||
|
|
||||||
|
### [0.1.20](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.19...v0.1.20) (2023-02-05)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* **cli:** :fire: add regex expression in addKeyworkd ([e34560c](https://github.com/leifermendez/bot-whatsapp/commit/e34560c77d4852d2e90930f0858e51aa67d4eeab))
|
||||||
|
* **provider:** :zap: possible get class provider ([76ba717](https://github.com/leifermendez/bot-whatsapp/commit/76ba717927a75b3d6299206aa0b8aee2bc25b726))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **cli:** :zap: working flowDynamic test ([c0113ca](https://github.com/leifermendez/bot-whatsapp/commit/c0113ca49295aff220d8defcb53f2ba7f2872d75))
|
||||||
|
* **cli:** :zap: working flowDynamic test ([aef52d2](https://github.com/leifermendez/bot-whatsapp/commit/aef52d2694fa6616d614338643db198b4f7f1fe8))
|
||||||
|
* **cli:** :zap: working flowDynamic test ([f769320](https://github.com/leifermendez/bot-whatsapp/commit/f76932021ce968d93241b55cfcdb8ae0e0e6c934))
|
||||||
|
* **cli:** :zap: working flowDynamic test ([23e09ef](https://github.com/leifermendez/bot-whatsapp/commit/23e09efaeccaf51018c55da492edff45b625f0a9))
|
||||||
|
* **database:** add support emoji in mysql ([9311aa0](https://github.com/leifermendez/bot-whatsapp/commit/9311aa0a65623a1bf40e96207a281625154dae90))
|
||||||
|
* **database:** fix naming ([cd082f2](https://github.com/leifermendez/bot-whatsapp/commit/cd082f235012cd5f5844c6437f51711beee0c865))
|
||||||
|
* **database:** fix naming ([1afc3ba](https://github.com/leifermendez/bot-whatsapp/commit/1afc3ba182070713b5bec40eaab0fa1f680830cd))
|
||||||
|
* **database:** fix naming ([c9831d2](https://github.com/leifermendez/bot-whatsapp/commit/c9831d202ab2c85f15a0247cd2a2426bc435270c))
|
||||||
|
* **provider:** :zap: baily wa.link ([96c2bff](https://github.com/leifermendez/bot-whatsapp/commit/96c2bffd093269be8e39474a84c156938504a6cb))
|
||||||
|
|
||||||
|
### [0.1.19](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.18...v0.1.19) (2023-01-29)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* :fire: bailey add media ([eab39e4](https://github.com/leifermendez/bot-whatsapp/commit/eab39e4ac06fd46f1a4671f8c15d1456b4400b97))
|
||||||
|
* :zap: more feature ([e19c3a2](https://github.com/leifermendez/bot-whatsapp/commit/e19c3a25a40259c74b4add9635af4844907eed26))
|
||||||
|
* **provider:** :rocket: fix issues in providers venom and wwebjs ([cbe438b](https://github.com/leifermendez/bot-whatsapp/commit/cbe438b77854e8df48b9dafaf7a837d21124ac5f))
|
||||||
|
* **provider:** :rocket: fix provider ([0ad4c58](https://github.com/leifermendez/bot-whatsapp/commit/0ad4c58457b548dc41c0f9e8470d59c48de7b95a))
|
||||||
|
* **provider:** :rocket: fix provider ([f8c7184](https://github.com/leifermendez/bot-whatsapp/commit/f8c7184487065443ab10f77aaf585e8bd63ca441))
|
||||||
|
* **provider:** :rocket: fix provider ([b2afa45](https://github.com/leifermendez/bot-whatsapp/commit/b2afa45352a7ab1f5d9775f3c1fde475bd8ca204))
|
||||||
|
* **provider:** :rocket: fix provider venom and wwebjs ([dcb0566](https://github.com/leifermendez/bot-whatsapp/commit/dcb0566d2bc3da40cd0c71554bb5ea0ec115d9ca))
|
||||||
|
* **provider:** :rocket: implements all send media to venom provider ([9dd7c02](https://github.com/leifermendez/bot-whatsapp/commit/9dd7c02b6a5474aff063f7d6be0ca8519504b93c))
|
||||||
|
* **provider:** :rocket: send file wwebjs ([6ff1a3a](https://github.com/leifermendez/bot-whatsapp/commit/6ff1a3a980196c01c66ed04ee07d0e7e57256504))
|
||||||
|
* **provider:** :zap: bailey add send file video audio ([14d1a61](https://github.com/leifermendez/bot-whatsapp/commit/14d1a61fa259c09135c37c55bd79e97c9c8367e4))
|
||||||
|
* **provider:** :zap: venom wweb ([fd2847a](https://github.com/leifermendez/bot-whatsapp/commit/fd2847aea0db17a0bdf33b5bca67a4cb8db2da16))
|
||||||
|
* **provider:** :zap: venom wweb ([f95331d](https://github.com/leifermendez/bot-whatsapp/commit/f95331d3dc70e76a3dfbe4c8d24059f0e7a164ef))
|
||||||
|
* **provider:** 🚀 implements all send media to venom provider ([bd7d150](https://github.com/leifermendez/bot-whatsapp/commit/bd7d150c047af41fdbb47f0a50a21e82cd79ee85))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **bot:** :fire: endFlow with ctx ([f6114af](https://github.com/leifermendez/bot-whatsapp/commit/f6114affadfbc324536a86167d1fdfe8da3c8de6))
|
||||||
|
* **bot:** :fire: endFlow with ctx ([b655ae4](https://github.com/leifermendez/bot-whatsapp/commit/b655ae449e7958ea940d8cc3c678fd66f60b6385))
|
||||||
|
* **bot:** :zap: endFlow butons ([87a4203](https://github.com/leifermendez/bot-whatsapp/commit/87a4203cd5b88f566387a76d586248e4265d6e4e))
|
||||||
|
* **bot:** :zap: fix fallback refactor ([e22780d](https://github.com/leifermendez/bot-whatsapp/commit/e22780d3faba94f71a70f1f201a20690608fa5bf))
|
||||||
|
* **cli:** :zap: endflow ([1c66f17](https://github.com/leifermendez/bot-whatsapp/commit/1c66f178a56d284bb8cb9df5ca17685c7e5d1ddd))
|
||||||
|
* **cli:** :zap: refactor fallback in child flow ([b33e346](https://github.com/leifermendez/bot-whatsapp/commit/b33e34692d3abcb6874308a9be79f74be4a2c3a8))
|
||||||
|
* **cli:** :zap: refactor fallback in child flow ([8da4b20](https://github.com/leifermendez/bot-whatsapp/commit/8da4b204b41125b5d0fa0aee4fa87c1f5faf5568))
|
||||||
|
|
||||||
|
### [0.1.18](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.17...v0.1.18) (2023-01-24)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* **bot:** :zap: add blacklist ([7078dc4](https://github.com/leifermendez/bot-whatsapp/commit/7078dc4c93d01bf90ef08ecb34e89a1abbe16fd2))
|
||||||
|
* **bot:** :zap: flowDynamic buttons, media ([3c4b1c0](https://github.com/leifermendez/bot-whatsapp/commit/3c4b1c0fc4b6d98d67c67806d918d3604bb2209b))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **bot:** :bug: body undefined ([bb6ed4a](https://github.com/leifermendez/bot-whatsapp/commit/bb6ed4a084ae98070dfdf0c4ba1eca574c4092cc))
|
||||||
|
* **bot:** :bug: body undefined ([9234cf1](https://github.com/leifermendez/bot-whatsapp/commit/9234cf1c5d00abdd35e62a826b3c450ab056987a))
|
||||||
|
* **bot:** :bug: body undefined ([a118bbb](https://github.com/leifermendez/bot-whatsapp/commit/a118bbbf7f0a7023cb7f33c23f37db72adad151f))
|
||||||
|
* **bot:** :bug: body undefined ([f54dea5](https://github.com/leifermendez/bot-whatsapp/commit/f54dea52b01063acd6122eeba1fbbe324aa7805d))
|
||||||
|
* **bot:** :bug: body undefined ([72e0a91](https://github.com/leifermendez/bot-whatsapp/commit/72e0a910503e9643db7dfbc6e09c41c96934e1f7))
|
||||||
|
* **bot:** :bug: body undefined ([70dd4d7](https://github.com/leifermendez/bot-whatsapp/commit/70dd4d73e814fc5636d19a887f3621c483b837c1))
|
||||||
|
* **bot:** :bug: body undefined ([ecf0eef](https://github.com/leifermendez/bot-whatsapp/commit/ecf0eef928917d76c59bd23886cb7a4108b421f1))
|
||||||
|
* **bot:** :bug: flowDynamic stranger behaviour ([877252b](https://github.com/leifermendez/bot-whatsapp/commit/877252bd4a8a7bbbbf083c3ceaeaeb952b0a1828))
|
||||||
|
* **bot:** :bug: flowDynamic stranger behaviour ([f5a7de3](https://github.com/leifermendez/bot-whatsapp/commit/f5a7de3a003c012e2164e51fff26892cfc3144be))
|
||||||
|
* **bot:** :memo: more docs ([98793d0](https://github.com/leifermendez/bot-whatsapp/commit/98793d0cfc1674830beaa3707f933c5a791eec14))
|
||||||
|
* **cli:** :zap: refactor ([a29b9d4](https://github.com/leifermendez/bot-whatsapp/commit/a29b9d4e1f85fc163cf1d633c0857f0c8b7f03e1))
|
||||||
|
* **cli:** :zap: refactor ([18ef4e9](https://github.com/leifermendez/bot-whatsapp/commit/18ef4e9d726575ca390ca24354825860328d3347))
|
||||||
|
* **cli:** :zap: refactor ([3648757](https://github.com/leifermendez/bot-whatsapp/commit/3648757fa083bdb88a16bf6c2e90c828c233bdb1))
|
||||||
|
* **cli:** :zap: refactor ([32f6a70](https://github.com/leifermendez/bot-whatsapp/commit/32f6a70f8f6fb26d8ea2a0f1a4aec4827b9d6a93))
|
||||||
|
* **cli:** :zap: refactor ([8c825e7](https://github.com/leifermendez/bot-whatsapp/commit/8c825e7f6b7133f7cc7f3041ce331b80a9fe60e0))
|
||||||
|
* **cli:** :zap: refactor ([0c0f437](https://github.com/leifermendez/bot-whatsapp/commit/0c0f4375b84549bee809340a85f9ce038ee2739e))
|
||||||
|
* **cli:** :zap: refactor ([039ce5d](https://github.com/leifermendez/bot-whatsapp/commit/039ce5dd7cac8115b335ad5de05f7bd871e24140))
|
||||||
|
* **cli:** :zap: refactor ([5e87918](https://github.com/leifermendez/bot-whatsapp/commit/5e879188b8bf9d486399b308a9a9c2612607d465))
|
||||||
|
* **cli:** :zap: refactor ([21a7270](https://github.com/leifermendez/bot-whatsapp/commit/21a72702817bc6b344223b34ca4513a7ff45fc93))
|
||||||
|
* **cli:** :zap: refactor ([82a99b2](https://github.com/leifermendez/bot-whatsapp/commit/82a99b2c80e6738566042ea738bbab8208a17758))
|
||||||
|
* **cli:** :zap: refactor ([cc19974](https://github.com/leifermendez/bot-whatsapp/commit/cc19974579379777b05cb69c38cec0fce6740471))
|
||||||
|
* **cli:** :zap: refactor ([56fcb8f](https://github.com/leifermendez/bot-whatsapp/commit/56fcb8fb72169bc21fce7c4fcdceccf2acd39c73))
|
||||||
|
* **cli:** :zap: refactor ([f36cff1](https://github.com/leifermendez/bot-whatsapp/commit/f36cff1eefdd96be4ab531e1cb2d3b630b1a81c3))
|
||||||
|
* **cli:** :zap: refactor ([b393c11](https://github.com/leifermendez/bot-whatsapp/commit/b393c11af6c0ebccb0a690be8b90b9df8877dad1))
|
||||||
|
* **cli:** :zap: refactor ([6683715](https://github.com/leifermendez/bot-whatsapp/commit/6683715ad617ea1075654a475a1c62ea607c733f))
|
||||||
|
* **contexts:** :bug: fixed [#524](https://github.com/leifermendez/bot-whatsapp/issues/524) issue ([79cc31a](https://github.com/leifermendez/bot-whatsapp/commit/79cc31a96f6a9836447cc4e6bb1e1521c54183fe))
|
||||||
|
* **contexts:** :bug: fixed [#524](https://github.com/leifermendez/bot-whatsapp/issues/524) issue ([7067b4a](https://github.com/leifermendez/bot-whatsapp/commit/7067b4a80b7938ccfaf1ed141a37d645a1a3a062))
|
||||||
|
* **provider:** wwebjs upgrade ([345f256](https://github.com/leifermendez/bot-whatsapp/commit/345f256a1b4a238519dafc15c9a31bc5e6bad4fe))
|
||||||
|
* se agrego @bot-whatsapp/portal a package.json ([46a9fa6](https://github.com/leifermendez/bot-whatsapp/commit/46a9fa6793e06600335de998d2bd9d0691b02ca4))
|
||||||
|
|
||||||
|
### [0.1.17](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.16...v0.1.17) (2023-01-13)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* mod de starters para habiltar portal ([eceb170](https://github.com/leifermendez/bot-whatsapp/commit/eceb170df03721dca4183b658c863b94fa04bc84))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **ci:** pre-release ([aaec075](https://github.com/leifermendez/bot-whatsapp/commit/aaec0751408ab49483d428810d94aaf7d46acb94))
|
||||||
|
* correccion en starters app.js para portal QR ([f430380](https://github.com/leifermendez/bot-whatsapp/commit/f430380b4f23d41702395c96c628bf13bf443278))
|
||||||
|
* **starters:** :zap: added dockerfile ([230981e](https://github.com/leifermendez/bot-whatsapp/commit/230981e2676361149cb2a99def7f705e75009260))
|
||||||
|
|
||||||
|
### [0.1.16](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.15...v0.1.16) (2023-01-11)
|
||||||
|
|
||||||
|
### [0.1.15](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.14...v0.1.15) (2023-01-11)
|
||||||
|
|
||||||
|
### [0.1.14](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.13...v0.1.14) (2023-01-11)
|
||||||
|
|
||||||
|
### [0.1.13](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.12...v0.1.13) (2023-01-11)
|
||||||
|
|
||||||
|
### [0.1.12](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.10...v0.1.12) (2023-01-11)
|
||||||
|
|
||||||
|
### [0.1.9-pre](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.9...v0.1.9-pre) (2023-01-10)
|
||||||
|
|
||||||
|
### [0.1.7-pre-1](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.7-alpha...v0.1.7-pre-1) (2023-01-10)
|
||||||
|
|
||||||
|
### [0.1.7-alpha](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.7-pre...v0.1.7-alpha) (2023-01-10)
|
||||||
|
|
||||||
|
### [0.1.11](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.10...v0.1.11) (2023-01-11)
|
||||||
|
|
||||||
|
### [0.1.9-pre](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.9...v0.1.9-pre) (2023-01-10)
|
||||||
|
|
||||||
|
### [0.1.7-pre-1](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.7-alpha...v0.1.7-pre-1) (2023-01-10)
|
||||||
|
|
||||||
|
### [0.1.7-alpha](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.7-pre...v0.1.7-alpha) (2023-01-10)
|
||||||
|
|
||||||
|
### [0.1.10](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.8...v0.1.10) (2023-01-11)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* :fire: update qr package ([ecde23f](https://github.com/leifermendez/bot-whatsapp/commit/ecde23fdea65def209aa874af35a3f293e6b1a91))
|
||||||
|
|
||||||
|
### [0.1.8](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.7-pre...v0.1.8) (2023-01-10)
|
||||||
|
|
||||||
|
### [0.1.7](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.7-pre...v0.1.7) (2023-01-10)
|
||||||
|
|
||||||
|
### [0.1.6](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.5...v0.1.6) (2023-01-10)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* :zap: new portal web for qr scan ([cb2e869](https://github.com/leifermendez/bot-whatsapp/commit/cb2e8692a3f94c8b24993cd11dd564f094b0e4ef))
|
||||||
|
* :zap: new portal web for qr scan ([9e93795](https://github.com/leifermendez/bot-whatsapp/commit/9e93795e6fce38890045389da95184fef1fbe0da))
|
||||||
|
* :zap: new portal web for qr scan ([3c178ea](https://github.com/leifermendez/bot-whatsapp/commit/3c178ea113b140535a51f5dcd521dbb66251670e))
|
||||||
|
* :zap: new portal web for qr scan ([1f1f564](https://github.com/leifermendez/bot-whatsapp/commit/1f1f564f4e2e3aa13b84de500fe215e0c45c2770))
|
||||||
|
* :zap: new portal web for qr scan ([3de5f4b](https://github.com/leifermendez/bot-whatsapp/commit/3de5f4b77a10e30632ff7555f5af5d8e93cb2019))
|
||||||
|
* :zap: qr code filename ([d794f60](https://github.com/leifermendez/bot-whatsapp/commit/d794f604ac8a835e523709dbf18c9b1609bbd00e))
|
||||||
|
* :zap: qr portal ([246ecdc](https://github.com/leifermendez/bot-whatsapp/commit/246ecdc11a8c4e652867c842b612dc4ce73f9828))
|
||||||
|
* :zap: qr portal ([af8b401](https://github.com/leifermendez/bot-whatsapp/commit/af8b401d075e1c35065589ede61476461ce86b4d))
|
||||||
|
* agregamos dockerfile y webserver a starters ([f9e3bbc](https://github.com/leifermendez/bot-whatsapp/commit/f9e3bbc6655060408e4fdbe1d7e920c2ed4fca53))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* :zap: add Dockerfile, starter ([4e0d33c](https://github.com/leifermendez/bot-whatsapp/commit/4e0d33c6bb46ad259774f6d0c38c6c0b5f8ca4a9))
|
||||||
|
* :zap: fix inject port args ([20f752e](https://github.com/leifermendez/bot-whatsapp/commit/20f752e6c1b1f7d11948fc4f2f8950f7834df7d9))
|
||||||
|
* :zap: fix inject port args ([7a23eb0](https://github.com/leifermendez/bot-whatsapp/commit/7a23eb0cc6f93ec21c5ab34e46981ae7a93f42ff))
|
||||||
|
* **provider:** :zap: fix send image baileys ([2ddea54](https://github.com/leifermendez/bot-whatsapp/commit/2ddea5468d235035478d4e91e63c821da19da179))
|
||||||
|
* **provider:** :zap: fix send image baileys ([391e11c](https://github.com/leifermendez/bot-whatsapp/commit/391e11ce738cd64792b5237d69f3739b0263c198))
|
||||||
|
* **provider:** :zap: fix send image baileys ([5d10cb9](https://github.com/leifermendez/bot-whatsapp/commit/5d10cb9026da60043e9a2f86117ebb04d0631a3f))
|
||||||
|
* **provider:** fix error docker as root user ([5a033da](https://github.com/leifermendez/bot-whatsapp/commit/5a033da83aee1f614120bccf27c9f330500cc7b0))
|
||||||
|
|
||||||
|
### [0.1.4](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.5...v0.1.4) (2023-01-10)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* :zap: new portal web for qr scan ([cb2e869](https://github.com/leifermendez/bot-whatsapp/commit/cb2e8692a3f94c8b24993cd11dd564f094b0e4ef))
|
||||||
|
* :zap: new portal web for qr scan ([9e93795](https://github.com/leifermendez/bot-whatsapp/commit/9e93795e6fce38890045389da95184fef1fbe0da))
|
||||||
|
* :zap: new portal web for qr scan ([3c178ea](https://github.com/leifermendez/bot-whatsapp/commit/3c178ea113b140535a51f5dcd521dbb66251670e))
|
||||||
|
* :zap: new portal web for qr scan ([1f1f564](https://github.com/leifermendez/bot-whatsapp/commit/1f1f564f4e2e3aa13b84de500fe215e0c45c2770))
|
||||||
|
* :zap: new portal web for qr scan ([3de5f4b](https://github.com/leifermendez/bot-whatsapp/commit/3de5f4b77a10e30632ff7555f5af5d8e93cb2019))
|
||||||
|
* :zap: qr code filename ([d794f60](https://github.com/leifermendez/bot-whatsapp/commit/d794f604ac8a835e523709dbf18c9b1609bbd00e))
|
||||||
|
* :zap: qr portal ([246ecdc](https://github.com/leifermendez/bot-whatsapp/commit/246ecdc11a8c4e652867c842b612dc4ce73f9828))
|
||||||
|
* :zap: qr portal ([af8b401](https://github.com/leifermendez/bot-whatsapp/commit/af8b401d075e1c35065589ede61476461ce86b4d))
|
||||||
|
* agregamos dockerfile y webserver a starters ([f9e3bbc](https://github.com/leifermendez/bot-whatsapp/commit/f9e3bbc6655060408e4fdbe1d7e920c2ed4fca53))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* :zap: add Dockerfile, starter ([4e0d33c](https://github.com/leifermendez/bot-whatsapp/commit/4e0d33c6bb46ad259774f6d0c38c6c0b5f8ca4a9))
|
||||||
|
* :zap: fix inject port args ([20f752e](https://github.com/leifermendez/bot-whatsapp/commit/20f752e6c1b1f7d11948fc4f2f8950f7834df7d9))
|
||||||
|
* :zap: fix inject port args ([7a23eb0](https://github.com/leifermendez/bot-whatsapp/commit/7a23eb0cc6f93ec21c5ab34e46981ae7a93f42ff))
|
||||||
|
* **provider:** :zap: fix send image baileys ([2ddea54](https://github.com/leifermendez/bot-whatsapp/commit/2ddea5468d235035478d4e91e63c821da19da179))
|
||||||
|
* **provider:** :zap: fix send image baileys ([391e11c](https://github.com/leifermendez/bot-whatsapp/commit/391e11ce738cd64792b5237d69f3739b0263c198))
|
||||||
|
* **provider:** :zap: fix send image baileys ([5d10cb9](https://github.com/leifermendez/bot-whatsapp/commit/5d10cb9026da60043e9a2f86117ebb04d0631a3f))
|
||||||
|
* **provider:** fix error docker as root user ([5a033da](https://github.com/leifermendez/bot-whatsapp/commit/5a033da83aee1f614120bccf27c9f330500cc7b0))
|
||||||
|
|
||||||
|
### [0.1.3](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.2...v0.1.3) (2023-01-04)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* **adapter:** :zap: send messages with dialogflow ([c20e151](https://github.com/leifermendez/bot-whatsapp/commit/c20e151e209d33de9e7425a64f003c85360f1832))
|
||||||
|
* **baileys:** added more methods ([1b23b83](https://github.com/leifermendez/bot-whatsapp/commit/1b23b837460ce4533ff33f10f1de5e3a344a5623))
|
||||||
|
* **bot:** :zap: http responses support ([e331c2d](https://github.com/leifermendez/bot-whatsapp/commit/e331c2dcc40eeb82a93f9d29f6a82333b8465927))
|
||||||
|
* **bot:** :zap: http responses support ([2d2bb08](https://github.com/leifermendez/bot-whatsapp/commit/2d2bb085cd95604a84ca3fe5c4ddc84b3824ac1c))
|
||||||
|
* **bot:** :zap: rev-03 everything work fine ([3012e02](https://github.com/leifermendez/bot-whatsapp/commit/3012e026b77ab4e99334b992d166a89189f76503))
|
||||||
|
* **cli:** :sparkles: added bailey ([06acec2](https://github.com/leifermendez/bot-whatsapp/commit/06acec2bf29d72c2b46f4ce81fed115bab97351f))
|
||||||
|
* **cli:** :sparkles: added bailey ([c868f73](https://github.com/leifermendez/bot-whatsapp/commit/c868f7346245bec94582b25a342febc657926c9d))
|
||||||
|
* **conflict:** :zap: remove unused variable ([eba9229](https://github.com/leifermendez/bot-whatsapp/commit/eba92299cfd84c971f09697d027043f19eec2b7c))
|
||||||
|
* **contexts:** :zap: add new dialogflowcx ([4d8cf62](https://github.com/leifermendez/bot-whatsapp/commit/4d8cf623ff86b3d08c8d52293d4e289dfda68e1c))
|
||||||
|
* **contexts:** :zap: add new dialogflowcx ([9885872](https://github.com/leifermendez/bot-whatsapp/commit/98858729919b2544dace07c49badce7888ddfd82))
|
||||||
|
* **contexts:** dialogflowcx support ([9179421](https://github.com/leifermendez/bot-whatsapp/commit/917942139f9736f1c0f8ce5f07b4e12e5768b0c7))
|
||||||
|
* correccion de flujos en app.js de ejemplo ([99f508f](https://github.com/leifermendez/bot-whatsapp/commit/99f508f93889d70240861158bc304c25a3b2daef))
|
||||||
|
* **docs:** master class updated ([69fd81a](https://github.com/leifermendez/bot-whatsapp/commit/69fd81a565e61b249ac50917585293d2d84e3dd4))
|
||||||
|
* **docs:** master class updated ([d522b03](https://github.com/leifermendez/bot-whatsapp/commit/d522b03e2e6e6e3f7c467c59e3d2d6f288fe37b2))
|
||||||
|
* **provider:** :bug: dialogflow ([4ec6f1e](https://github.com/leifermendez/bot-whatsapp/commit/4ec6f1e120879e545fa111615f2d79b792d947a5))
|
||||||
|
* **provider:** :sparkles: added dialogflow ([2f633c7](https://github.com/leifermendez/bot-whatsapp/commit/2f633c72da24f98d6c318d1e725571b62e04604c))
|
||||||
|
* **provider:** :sparkles: added dialogflow ([798f1ce](https://github.com/leifermendez/bot-whatsapp/commit/798f1cebdefe43624c1698a219dcb224bb842d38))
|
||||||
|
* **provider:** :sparkles: endpoint is added to validate the webhook … ([478929d](https://github.com/leifermendez/bot-whatsapp/commit/478929d1340d46d6bf997ae8edabbaae4511172d))
|
||||||
|
* **provider:** :sparkles: endpoint is added to validate the webhook token ([1ec1564](https://github.com/leifermendez/bot-whatsapp/commit/1ec15647dc462363d5b765f42debddbe6ef6266b))
|
||||||
|
* **provider:** :zap: add new methods ([0b4e353](https://github.com/leifermendez/bot-whatsapp/commit/0b4e35308dace0ccdf618cb1d04987ed5200d58c))
|
||||||
|
* **provider:** :zap: add sendfile and sendButtons ([5433610](https://github.com/leifermendez/bot-whatsapp/commit/5433610a84d7a050a387e4daf2ded1daebfc03a4))
|
||||||
|
* **provider:** :zap: add sendfile and sendButtons ([342cbcc](https://github.com/leifermendez/bot-whatsapp/commit/342cbccff1d09f9aabe5423ad6d686d590a2448f))
|
||||||
|
* **provider:** :zap: added new venom provider ([01fe9eb](https://github.com/leifermendez/bot-whatsapp/commit/01fe9ebc9a943f2aa086ee415153d1cccdb14ec0))
|
||||||
|
* **provider:** :zap: added tamplate venom ([337c2e9](https://github.com/leifermendez/bot-whatsapp/commit/337c2e94bccd0ae173958fe2db08b494bdc93c28))
|
||||||
|
* **provider:** :zap: baileysProvider ([23b2e8e](https://github.com/leifermendez/bot-whatsapp/commit/23b2e8e439ecec24450bd5cf1a3820316e643434))
|
||||||
|
* **provider:** :zap: solution error buttons ([1b83871](https://github.com/leifermendez/bot-whatsapp/commit/1b83871cca6996c6acae3d4c8b6b42aec05ea146))
|
||||||
|
* **provider:** :zap: solution error utils venom ([31c83f5](https://github.com/leifermendez/bot-whatsapp/commit/31c83f5d689a01490d3adb96006f54c2a5d3268b))
|
||||||
|
* **provider:** :zap: update ([b62d21a](https://github.com/leifermendez/bot-whatsapp/commit/b62d21a0bf94466e43c25c6e8c0f5db9ae91c572))
|
||||||
|
* **provider:** :zap: update ([0c94647](https://github.com/leifermendez/bot-whatsapp/commit/0c94647a27747c3ddf4f02926580370f0d81bdc2))
|
||||||
|
* **provider:** meta provider is added ([b041f7d](https://github.com/leifermendez/bot-whatsapp/commit/b041f7d0c7cc6f152d3f36785d1d398a4141d57d))
|
||||||
|
* **provider:** meta provider is added ([438607c](https://github.com/leifermendez/bot-whatsapp/commit/438607c222b91d6f8814201dabe5f7c3e7ba1abb))
|
||||||
|
* **provider:** new added baileys ([4e0fcbd](https://github.com/leifermendez/bot-whatsapp/commit/4e0fcbd8347f8a430adb43351b5415098a5d10df))
|
||||||
|
* **provider:** new provider - venon:zap: configuracion inicial provi… ([66f75f8](https://github.com/leifermendez/bot-whatsapp/commit/66f75f872200334bfc9eda744bed92c509dfee56))
|
||||||
|
* **provider:** new provider - venon:zap: configuracion inicial provider venom ([fee7c2e](https://github.com/leifermendez/bot-whatsapp/commit/fee7c2e967b7fe8835b5acc243c19f7713acfbe7))
|
||||||
|
* se agregaron los datros del adapter mongo en app.js y package.json ([8160d13](https://github.com/leifermendez/bot-whatsapp/commit/8160d13c866b8ae17b0ec8e68eee1bc0373595b0))
|
||||||
|
* se agrego informacion al ejemplo en app.js ([954e751](https://github.com/leifermendez/bot-whatsapp/commit/954e751f700c6a39ec70c0bc5168637c0dc7e07c))
|
||||||
|
* se agrego informacion al ejemplo en app.js ([b2f1339](https://github.com/leifermendez/bot-whatsapp/commit/b2f13396104db9ccef5b3bad1c4e19c6a4bad2d4))
|
||||||
|
* **starters:** meta memory base template added ([11c784f](https://github.com/leifermendez/bot-whatsapp/commit/11c784f882965d6bd3a2313cf91bed9fb3aa5f26))
|
||||||
|
* **starters:** meta memory base template added ([e8d6252](https://github.com/leifermendez/bot-whatsapp/commit/e8d625201ed86e162e0b4e82100ede1d08985555))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* :art: update ([7d6708c](https://github.com/leifermendez/bot-whatsapp/commit/7d6708c01bbdc5043a7e6ed56fe15a9618115b91))
|
||||||
|
* :sparkles: updated starters ([5da4b7a](https://github.com/leifermendez/bot-whatsapp/commit/5da4b7a4d1e5950be94361ac439938741b9d299c))
|
||||||
|
* actualizar app.js de ejemplo ([1746613](https://github.com/leifermendez/bot-whatsapp/commit/17466138ddcef60a23a0c87911f22045f26d3233))
|
||||||
|
* actualizar ejemplo app.js ([60fdbf3](https://github.com/leifermendez/bot-whatsapp/commit/60fdbf3d3cd62819e618853a9dc2fd0e23fe8752))
|
||||||
|
* **adapter:** :fire: clear log ([9ad4874](https://github.com/leifermendez/bot-whatsapp/commit/9ad4874fdafabfbf0e9e20e6b3281f702bb9fbe7))
|
||||||
|
* **adapter:** :fire: clear log ([4d34d3a](https://github.com/leifermendez/bot-whatsapp/commit/4d34d3ab1daab4e72fb5244216c78cf836d1a164))
|
||||||
|
* **adapter:** :fire: improvement baileys ([e6fefb4](https://github.com/leifermendez/bot-whatsapp/commit/e6fefb4049847f996f2a169b9acfc27c2428d3e6))
|
||||||
|
* **adapter:** :fire: improvement baileys ([2d5ac26](https://github.com/leifermendez/bot-whatsapp/commit/2d5ac2664bea09e60ac85ff2612609ae21050945))
|
||||||
|
* **adapter:** :rocket: venom update - cli - qr iamge ([041bf62](https://github.com/leifermendez/bot-whatsapp/commit/041bf6280e5f6956393716907e0669aa3ca78b4a))
|
||||||
|
* **adapter:** :rocket: venom update - cli - qr iamge ([e37fd0d](https://github.com/leifermendez/bot-whatsapp/commit/e37fd0da3635aa1041664d490d5f9803d2c441ca))
|
||||||
|
* **adapter:** :rocket: venom update - cli - qr iamge ([ca6afbb](https://github.com/leifermendez/bot-whatsapp/commit/ca6afbb87fceec12d4a383486ad693905e36881f))
|
||||||
|
* **adapter:** json db change is made ([386c1bb](https://github.com/leifermendez/bot-whatsapp/commit/386c1bbbac036aa58335fb5f62e3af2493766b6b))
|
||||||
|
* **adapter:** json db change is made ([3bdc7af](https://github.com/leifermendez/bot-whatsapp/commit/3bdc7afe8062527ff08620650d2c1177dfea83f5))
|
||||||
|
* agregamos variables para mysql ([dcf65b8](https://github.com/leifermendez/bot-whatsapp/commit/dcf65b87bc7e7e6381e6448e83118077986898e7))
|
||||||
|
* **bot:** :ambulance: fix callback functions ([d9aa97c](https://github.com/leifermendez/bot-whatsapp/commit/d9aa97c7819aca1446657bc0b75e9732f0f20c6b)), closes [#252](https://github.com/leifermendez/bot-whatsapp/issues/252)
|
||||||
|
* **bot:** :ambulance: fix callback functions ([964a074](https://github.com/leifermendez/bot-whatsapp/commit/964a074aa41324bd09d0c4e2e7aa663a0602b69c))
|
||||||
|
* **bot:** :fire: fix rev ([21407c0](https://github.com/leifermendez/bot-whatsapp/commit/21407c0e37f1ab12efecf887e699cedf05e3946a))
|
||||||
|
* **bot:** :fire: fix rev ([484c8c3](https://github.com/leifermendez/bot-whatsapp/commit/484c8c3bdefbc7824c32a86090bafae0593ecdac))
|
||||||
|
* **bot:** :zap: working callback Phase 1 ([952ce86](https://github.com/leifermendez/bot-whatsapp/commit/952ce86ffaa48a0d6fbc0a00a08c5d1efa14ee8e))
|
||||||
|
* **bot:** :zap: working nested new flow ([2cbc962](https://github.com/leifermendez/bot-whatsapp/commit/2cbc96245d795de749d894a3a0d99b6550f08d9e))
|
||||||
|
* **cli:** :art: starters ([a2be57f](https://github.com/leifermendez/bot-whatsapp/commit/a2be57f0aa42c6b5e13ad19c34abc7d9e81dc135))
|
||||||
|
* **cli:** :art: starters ([670ecf1](https://github.com/leifermendez/bot-whatsapp/commit/670ecf121babf53e76c2ea106c0710cbe59facde))
|
||||||
|
* **cli:** :fire: update instructions ([e585e2f](https://github.com/leifermendez/bot-whatsapp/commit/e585e2f5f644ed0188dc9cd2b3c697c9d6050669))
|
||||||
|
* **cli:** :fire: update instructions ([ed36ce0](https://github.com/leifermendez/bot-whatsapp/commit/ed36ce0a7796320c6a4a452f29c05a3f0f7368db))
|
||||||
|
* **cli:** :fire: update instructions ([bad1694](https://github.com/leifermendez/bot-whatsapp/commit/bad16943fc2089887d6bf0b6d90075d3bec6f9c7))
|
||||||
|
* **cli:** :fire: update instructions ([a21633f](https://github.com/leifermendez/bot-whatsapp/commit/a21633fb7cf348cc37f4e4714f51172b49b193b5))
|
||||||
|
* **cli:** :zap: updated ([a6f4aa8](https://github.com/leifermendez/bot-whatsapp/commit/a6f4aa8d1e809330c06c165aaf9a9f90b8922bb5))
|
||||||
|
* **conflict:** conflict resolution ([71d43b5](https://github.com/leifermendez/bot-whatsapp/commit/71d43b585a0ce173061c84e9879915e4602db026))
|
||||||
|
* **contexts:** :fire: added buttons ([eabef7a](https://github.com/leifermendez/bot-whatsapp/commit/eabef7a92d005cd0190196cfe75828c38885aadf))
|
||||||
|
* **contexts:** :fire: added buttons ([1b878d2](https://github.com/leifermendez/bot-whatsapp/commit/1b878d2ba0daeb3609af74a2ebae7948456e7fb0))
|
||||||
|
* **contexts:** :fire: added buttons ([78b0a9d](https://github.com/leifermendez/bot-whatsapp/commit/78b0a9dddc2a6e0fceb721ee7794efa2047f25fc))
|
||||||
|
* **contexts:** :fire: added buttons ([d8309f7](https://github.com/leifermendez/bot-whatsapp/commit/d8309f77e1d9137c0bec977ed9faef633cd90552))
|
||||||
|
* correccion en app.js para remover addChild en starters ([32db429](https://github.com/leifermendez/bot-whatsapp/commit/32db429f2946f344d949cb169a9595d657c06279))
|
||||||
|
* fix del db provider mysql ([b59d4fc](https://github.com/leifermendez/bot-whatsapp/commit/b59d4fcdd7462cde3f68ab5746d49960b547a592))
|
||||||
|
* provider equivocado en app.js de venom ([4e0a109](https://github.com/leifermendez/bot-whatsapp/commit/4e0a1091ee85cedfaa5a9c3d40e5cd50bc36cda3))
|
||||||
|
* **provider:** :bug: create static site html qr ([c7e56a4](https://github.com/leifermendez/bot-whatsapp/commit/c7e56a4b13c8829f91769eeca7f1f6b3473f68cf))
|
||||||
|
* **provider:** :bug: fix metea provider ([85f50be](https://github.com/leifermendez/bot-whatsapp/commit/85f50be9dcbf3817107898d8d2980baf05acd678))
|
||||||
|
* **provider:** :bug: fix metea provider ([a52aaa1](https://github.com/leifermendez/bot-whatsapp/commit/a52aaa11d883bbaf526cf87720d3c3fd9f89a986))
|
||||||
|
* **provider:** :bug: qr code accurate ([6c4845d](https://github.com/leifermendez/bot-whatsapp/commit/6c4845d733720d9916bb4008f9069ae4fd986a4b))
|
||||||
|
* **provider:** :bug: qr code accurate ([91bfdc4](https://github.com/leifermendez/bot-whatsapp/commit/91bfdc46301207cbc5274308da6f39c7b4652c63))
|
||||||
|
* **provider:** :fire: baileys fix ([928365d](https://github.com/leifermendez/bot-whatsapp/commit/928365dcafb3631acf6b1d0c239a906f8e1c4b0d))
|
||||||
|
* **provider:** :fire: send message togther with media ([78aa23f](https://github.com/leifermendez/bot-whatsapp/commit/78aa23fab094059145f82e6781f9366d5d582b4f))
|
||||||
|
* **provider:** :fire: send message togther with media ([b6bf43d](https://github.com/leifermendez/bot-whatsapp/commit/b6bf43d70fc28c6a229522b9b0de76cec43ac864))
|
||||||
|
* **provider:** :zap: baileys fix restart ([ae83774](https://github.com/leifermendez/bot-whatsapp/commit/ae83774365027e2e86127ab7713ae9ee2df31f33))
|
||||||
|
* **provider:** :zap: edit starter ([ff65832](https://github.com/leifermendez/bot-whatsapp/commit/ff65832012003423cc86d25cf0923452b1f8acb7))
|
||||||
|
* **provider:** :zap: edit starter ([68dd182](https://github.com/leifermendez/bot-whatsapp/commit/68dd1820f05d04780824b318072d053eaf7db654))
|
||||||
|
* **provider:** :zap: json space ([3cef741](https://github.com/leifermendez/bot-whatsapp/commit/3cef741c9ee30024eb42770a5f32931fcd372160))
|
||||||
|
* **provider:** :zap: json space ([9b087e0](https://github.com/leifermendez/bot-whatsapp/commit/9b087e071019a7b6c79195a24dc7ddec498c5716))
|
||||||
|
* **provider:** :zap: json space ([208fb4e](https://github.com/leifermendez/bot-whatsapp/commit/208fb4e9131dd5d4fd7230ba1aa11181337d9181))
|
||||||
|
* **provider:** :zap: json space ([54a59c7](https://github.com/leifermendez/bot-whatsapp/commit/54a59c7f0d4dbaab006ce7e3c74412d8d3613ecd))
|
||||||
|
* **provider:** qr-fix margin ([663641a](https://github.com/leifermendez/bot-whatsapp/commit/663641a1b8bf9234a88b0f3c38381ebc4bfa4bf9))
|
||||||
|
* se quito addChild de las constantes porque no se usa ([ba2291a](https://github.com/leifermendez/bot-whatsapp/commit/ba2291a3ddac0d4101021e11d03cb222c5a4bb3b))
|
||||||
|
* **starters:** :fire: updated staters ([4d4f15c](https://github.com/leifermendez/bot-whatsapp/commit/4d4f15ce73486d9335ad474d9e37c3b155670134))
|
||||||
|
* **starters:** :fire: updated staters ([a30eaac](https://github.com/leifermendez/bot-whatsapp/commit/a30eaac77534d17eb980f6ec126140e9d30aa06e))
|
||||||
|
* **starters:** :memo: update MIGRATION ([37fe323](https://github.com/leifermendez/bot-whatsapp/commit/37fe32322eb1bd41eecd151e52f17ec0588fb85e))
|
||||||
|
* **starters:** :memo: update MIGRATION ([9b30e7d](https://github.com/leifermendez/bot-whatsapp/commit/9b30e7dcfc30bc160b56427cc6cdc2dc982bde2a))
|
||||||
|
* **starters:** base templates are added for meta ([229e017](https://github.com/leifermendez/bot-whatsapp/commit/229e017ae20b84c9d12c7282f97b7034f5f33e6d))
|
||||||
|
* **starters:** base templates are added for meta ([20f6651](https://github.com/leifermendez/bot-whatsapp/commit/20f665175c9b47226df41ce43e05574bd6ab1930))
|
||||||
|
|
||||||
|
### [0.1.2](https://github.com/leifermendez/bot-whatsapp/compare/v0.1.1...v0.1.2) (2022-12-12)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **cli:** :art: starters ([79e2318](https://github.com/leifermendez/bot-whatsapp/commit/79e231825613f33bfec2ae8e93139f885c199c7a))
|
||||||
|
* **cli:** :art: starters ([87ba43a](https://github.com/leifermendez/bot-whatsapp/commit/87ba43a5535be0893a7701a3b6a085ee5d29e7c5))
|
||||||
|
* fix dependencias ([61d0324](https://github.com/leifermendez/bot-whatsapp/commit/61d032426119341187a470035d49b8b252ca46cd))
|
||||||
|
|
||||||
|
### [0.1.1](https://github.com/leifermendez/bot-whatsapp/compare/v0.3.0...v0.1.1) (2022-12-12)
|
||||||
|
|
||||||
|
## 0.3.0 (2022-12-12)
|
||||||
|
|
||||||
|
|
||||||
|
### ⚠ BREAKING CHANGES
|
||||||
|
|
||||||
|
* 🧨 NO
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* (🎸) add onClick prop to component ([4ae3898](https://github.com/leifermendez/bot-whatsapp/commit/4ae389846d38c133f6bb2129ae373eed39d9d08d))
|
||||||
|
* **adapter:** added adapter mysql ([717a7dc](https://github.com/leifermendez/bot-whatsapp/commit/717a7dc95fbc107ec6f55387aff606c47144baa4))
|
||||||
|
* **adapter:** implementation of json file adapter ([5e1a373](https://github.com/leifermendez/bot-whatsapp/commit/5e1a3737303c843095984f6357564ea18458362f))
|
||||||
|
* **adapter:** mysql adapter ([8d73c86](https://github.com/leifermendez/bot-whatsapp/commit/8d73c86946d07aa80e5b375b62b84a88b2892e03))
|
||||||
|
* **adapter:** sql is added to create the table ([4b7de0f](https://github.com/leifermendez/bot-whatsapp/commit/4b7de0f6901524fa2c09271c3a99c364e6b3c260))
|
||||||
|
* **bot:** :fire: improvement provider handler ([4154cc2](https://github.com/leifermendez/bot-whatsapp/commit/4154cc223091a46d3203d3a378cd42f61749a5fa))
|
||||||
|
* **bot:** :zap: add send image function ([ce8a96b](https://github.com/leifermendez/bot-whatsapp/commit/ce8a96b958ff814c65d8fb4dbd5eaed5cc83a1ed))
|
||||||
|
* **ci:** :art: add releases ([b115dc3](https://github.com/leifermendez/bot-whatsapp/commit/b115dc3654996f049837bfb6b5d039a2313df0ad))
|
||||||
|
* **ci:** :art: relases script ([2e906bc](https://github.com/leifermendez/bot-whatsapp/commit/2e906bce79f7d854d437334e1d0c2cd270b0bbc6))
|
||||||
|
* **ci:** update ci ([e5a9db7](https://github.com/leifermendez/bot-whatsapp/commit/e5a9db7e12340c4f5baa66e8b20585b63daa3bcf))
|
||||||
|
* **cli:** create-starter ([3977987](https://github.com/leifermendez/bot-whatsapp/commit/397798790ef5857ca758b0df8384c6a4bfacc181))
|
||||||
|
* **provider:** :fire: add twilii (weoking) ([4350dff](https://github.com/leifermendez/bot-whatsapp/commit/4350dff22a7de69ba6d35ecbdd67e59b810bd46f))
|
||||||
|
* **provider:** added twilio provider ([8dd3be9](https://github.com/leifermendez/bot-whatsapp/commit/8dd3be909b36717f6b54e141a5f48d2722d4855c))
|
||||||
|
* **release:** added ([1988948](https://github.com/leifermendez/bot-whatsapp/commit/1988948c30d922beb7b83faab96d1d59cf7f5f90))
|
||||||
|
* **release:** added ([f4ad704](https://github.com/leifermendez/bot-whatsapp/commit/f4ad7040abf619635480c30babd6f1159c7af85a))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **adapter:** conflict resolution ([4b307ef](https://github.com/leifermendez/bot-whatsapp/commit/4b307efe79c738a5c4e04ff1c07ca247d827593c))
|
||||||
|
* **adapter:** corrections are made to the adapter ([afa6771](https://github.com/leifermendez/bot-whatsapp/commit/afa677190392d48715930ebe2b1e15c7619d730f))
|
||||||
|
* **bot:** :zap: added delay promises ([73caf09](https://github.com/leifermendez/bot-whatsapp/commit/73caf090ba9013132e5dcb7761a10939dc9ac300))
|
||||||
|
* **bot:** :zap: fix sensitive case ([24ac9fb](https://github.com/leifermendez/bot-whatsapp/commit/24ac9fbf48f80eeb521a36bc938af3a70dd82303))
|
||||||
|
* **bot:** :zap: flow improvement + add utils ([a7b19d9](https://github.com/leifermendez/bot-whatsapp/commit/a7b19d9bff5ea66ff888555c3df37ae0e20b612a))
|
||||||
|
* **bot:** update ([49698bf](https://github.com/leifermendez/bot-whatsapp/commit/49698bfda9d2a53f7b1a7e1724a796698601fbaa))
|
||||||
|
* **ci:** :zap: balance version ([ec46cfd](https://github.com/leifermendez/bot-whatsapp/commit/ec46cfdd657c08c8e90261613f00cfc080f1e1d6))
|
||||||
|
* **ci:** :zap: github action ([b827a0a](https://github.com/leifermendez/bot-whatsapp/commit/b827a0ab225b89bb8117c82628db0679c09b4102))
|
||||||
|
* **ci:** :zap: github action ([4142ca4](https://github.com/leifermendez/bot-whatsapp/commit/4142ca4fd552e7005f3b1397a76b90a2e574d19d))
|
||||||
|
* **ci:** :zap: github action ([091544a](https://github.com/leifermendez/bot-whatsapp/commit/091544ac3fac0c16925e856e1aec64bcad0ecf6d))
|
||||||
|
* **ci:** :zap: github action ([2ce342a](https://github.com/leifermendez/bot-whatsapp/commit/2ce342a0cb15019d5084ca06dc30e342b030ea10))
|
||||||
|
* **ci:** :zap: github action ([7817793](https://github.com/leifermendez/bot-whatsapp/commit/781779328f93ef8b0e6e0f85c6cd05ae782112fb))
|
||||||
|
* **ci:** :zap: github action ([dfced8c](https://github.com/leifermendez/bot-whatsapp/commit/dfced8c594e9175c81e837af359631ba055b7e1a))
|
||||||
|
* **ci:** :zap: github action ([aaa4ce8](https://github.com/leifermendez/bot-whatsapp/commit/aaa4ce837229fd51e274de3d91e1d9d615ac69fd))
|
||||||
|
* **ci:** :zap: github action ([9ddf144](https://github.com/leifermendez/bot-whatsapp/commit/9ddf144244cd6877e7d26f576387814459f2befb))
|
||||||
|
* **ci:** :zap: github action ([b465de5](https://github.com/leifermendez/bot-whatsapp/commit/b465de55a0e511213d1a7760a74efa102172c85e))
|
||||||
|
* **ci:** :zap: github action ([cf1dc6f](https://github.com/leifermendez/bot-whatsapp/commit/cf1dc6fac810545e5a2b63f31f71322f37329e38))
|
||||||
|
* **ci:** :zap: github action ([8d897f8](https://github.com/leifermendez/bot-whatsapp/commit/8d897f824e27a55ca011163092a813a7e8f426af))
|
||||||
|
* **ci:** ci ([f55cfae](https://github.com/leifermendez/bot-whatsapp/commit/f55cfae6e4ccc1df949212999406680020d27f9c))
|
||||||
|
* **ci:** ci ([671c5b3](https://github.com/leifermendez/bot-whatsapp/commit/671c5b37f33360e8cb754625b8dd6e83bce9014d))
|
||||||
|
* **cli:** :bug: path ([32212fb](https://github.com/leifermendez/bot-whatsapp/commit/32212fb52d206bf6f8d753a86d9ce40aa0db2a5d))
|
||||||
|
* **cli:** :fire: create script - templates ([2319db3](https://github.com/leifermendez/bot-whatsapp/commit/2319db3009501fe57ae21e60ad286eb68c46f4fd))
|
||||||
|
* **cli:** :fire: create script - templates ([9cb98b5](https://github.com/leifermendez/bot-whatsapp/commit/9cb98b5e73fca3c3f5e70a8497badc31e494b943))
|
||||||
|
* **cli:** :fire: create script - templates ([2999e0e](https://github.com/leifermendez/bot-whatsapp/commit/2999e0e753f31a8b9e6d7c117e78cdb5656e203a))
|
||||||
|
* **cli:** :fire: create script - templates ([af716b7](https://github.com/leifermendez/bot-whatsapp/commit/af716b75372899877a81b528b58278376166d0ad))
|
||||||
|
* **cli:** :fire: create script - templates ([c6999c8](https://github.com/leifermendez/bot-whatsapp/commit/c6999c84931083a87b5717db58003be68244707e))
|
||||||
|
* **cli:** :fire: create script - templates ([d4b49a9](https://github.com/leifermendez/bot-whatsapp/commit/d4b49a9bd7085070f0c5964d2903f10b71bde0b3))
|
||||||
|
* **cli:** :fire: create script - templates ([eebc3c9](https://github.com/leifermendez/bot-whatsapp/commit/eebc3c980638d88f11a0d93b8344f3ff345c7ee5))
|
||||||
|
* **cli:** :zap: clean eslinter ([bfb69d9](https://github.com/leifermendez/bot-whatsapp/commit/bfb69d9a9574a757ae02748b6c5f5afa3eac68e6))
|
||||||
|
* **cli:** :zap: clean eslinter ([15f6972](https://github.com/leifermendez/bot-whatsapp/commit/15f697225775a0f0e0a440cd980f7fb8f51a1056))
|
||||||
|
* **cli:** :zap: create-starter ([d3b8310](https://github.com/leifermendez/bot-whatsapp/commit/d3b8310180d2ad813733b1d18f2c32d7d947740a))
|
||||||
|
* **cli:** :zap: update cli copy ([7797c2b](https://github.com/leifermendez/bot-whatsapp/commit/7797c2b46133697e2a591adab2b67e66b34a1cfe))
|
||||||
|
* **fix:** fix ([6483545](https://github.com/leifermendez/bot-whatsapp/commit/648354500b123f20044f5ac2e8a26b15f16d1b8d))
|
||||||
|
* **fix:** fix ([28c0480](https://github.com/leifermendez/bot-whatsapp/commit/28c0480b8bfa6b24394095f57c36ef89c9aeb566))
|
||||||
|
* **linter:** update linter and commitlint ([70a94ab](https://github.com/leifermendez/bot-whatsapp/commit/70a94ab2c6f8e4122780c77bc3a621944883e621))
|
||||||
|
* pre-copy fix ([08e2552](https://github.com/leifermendez/bot-whatsapp/commit/08e2552907c48cfeaac843457a18bf2032e6f8aa))
|
||||||
|
* pre-copy fix ([6617107](https://github.com/leifermendez/bot-whatsapp/commit/6617107ab824215c449e26eae6c2bb327ecfc092))
|
||||||
|
* **starter:** pre-copy fix ([929e74c](https://github.com/leifermendez/bot-whatsapp/commit/929e74c84b667ec13cb5490b3b951cb8df15ebd1))
|
||||||
|
|
||||||
|
|
||||||
|
* (💍) Is justa test! ([37d04e9](https://github.com/leifermendez/bot-whatsapp/commit/37d04e9e89d3f01fdc367654ba60fb11ab2614c4))
|
||||||
|
|
||||||
|
### [0.2.1](https://github.com/leifermendez/bot-whatsapp/compare/v0.2.0...v0.2.1) (2022-12-12)
|
||||||
|
|
||||||
|
## 0.2.0 (2022-12-12)
|
||||||
|
|
||||||
|
|
||||||
|
### ⚠ BREAKING CHANGES
|
||||||
|
|
||||||
|
* 🧨 NO
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* (🎸) add onClick prop to component ([4ae3898](https://github.com/leifermendez/bot-whatsapp/commit/4ae389846d38c133f6bb2129ae373eed39d9d08d))
|
||||||
|
* **adapter:** added adapter mysql ([717a7dc](https://github.com/leifermendez/bot-whatsapp/commit/717a7dc95fbc107ec6f55387aff606c47144baa4))
|
||||||
|
* **adapter:** implementation of json file adapter ([5e1a373](https://github.com/leifermendez/bot-whatsapp/commit/5e1a3737303c843095984f6357564ea18458362f))
|
||||||
|
* **adapter:** mysql adapter ([8d73c86](https://github.com/leifermendez/bot-whatsapp/commit/8d73c86946d07aa80e5b375b62b84a88b2892e03))
|
||||||
|
* **adapter:** sql is added to create the table ([4b7de0f](https://github.com/leifermendez/bot-whatsapp/commit/4b7de0f6901524fa2c09271c3a99c364e6b3c260))
|
||||||
|
* **bot:** :fire: improvement provider handler ([4154cc2](https://github.com/leifermendez/bot-whatsapp/commit/4154cc223091a46d3203d3a378cd42f61749a5fa))
|
||||||
|
* **bot:** :zap: add send image function ([ce8a96b](https://github.com/leifermendez/bot-whatsapp/commit/ce8a96b958ff814c65d8fb4dbd5eaed5cc83a1ed))
|
||||||
|
* **ci:** :art: add releases ([b115dc3](https://github.com/leifermendez/bot-whatsapp/commit/b115dc3654996f049837bfb6b5d039a2313df0ad))
|
||||||
|
* **ci:** :art: relases script ([2e906bc](https://github.com/leifermendez/bot-whatsapp/commit/2e906bce79f7d854d437334e1d0c2cd270b0bbc6))
|
||||||
|
* **ci:** update ci ([e5a9db7](https://github.com/leifermendez/bot-whatsapp/commit/e5a9db7e12340c4f5baa66e8b20585b63daa3bcf))
|
||||||
|
* **cli:** create-starter ([3977987](https://github.com/leifermendez/bot-whatsapp/commit/397798790ef5857ca758b0df8384c6a4bfacc181))
|
||||||
|
* **provider:** :fire: add twilii (weoking) ([4350dff](https://github.com/leifermendez/bot-whatsapp/commit/4350dff22a7de69ba6d35ecbdd67e59b810bd46f))
|
||||||
|
* **provider:** added twilio provider ([8dd3be9](https://github.com/leifermendez/bot-whatsapp/commit/8dd3be909b36717f6b54e141a5f48d2722d4855c))
|
||||||
|
* **release:** added ([1988948](https://github.com/leifermendez/bot-whatsapp/commit/1988948c30d922beb7b83faab96d1d59cf7f5f90))
|
||||||
|
* **release:** added ([f4ad704](https://github.com/leifermendez/bot-whatsapp/commit/f4ad7040abf619635480c30babd6f1159c7af85a))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **adapter:** conflict resolution ([4b307ef](https://github.com/leifermendez/bot-whatsapp/commit/4b307efe79c738a5c4e04ff1c07ca247d827593c))
|
||||||
|
* **adapter:** corrections are made to the adapter ([afa6771](https://github.com/leifermendez/bot-whatsapp/commit/afa677190392d48715930ebe2b1e15c7619d730f))
|
||||||
|
* **bot:** :zap: added delay promises ([73caf09](https://github.com/leifermendez/bot-whatsapp/commit/73caf090ba9013132e5dcb7761a10939dc9ac300))
|
||||||
|
* **bot:** :zap: fix sensitive case ([24ac9fb](https://github.com/leifermendez/bot-whatsapp/commit/24ac9fbf48f80eeb521a36bc938af3a70dd82303))
|
||||||
|
* **bot:** :zap: flow improvement + add utils ([a7b19d9](https://github.com/leifermendez/bot-whatsapp/commit/a7b19d9bff5ea66ff888555c3df37ae0e20b612a))
|
||||||
|
* **bot:** update ([49698bf](https://github.com/leifermendez/bot-whatsapp/commit/49698bfda9d2a53f7b1a7e1724a796698601fbaa))
|
||||||
|
* **ci:** :zap: balance version ([ec46cfd](https://github.com/leifermendez/bot-whatsapp/commit/ec46cfdd657c08c8e90261613f00cfc080f1e1d6))
|
||||||
|
* **ci:** :zap: github action ([b827a0a](https://github.com/leifermendez/bot-whatsapp/commit/b827a0ab225b89bb8117c82628db0679c09b4102))
|
||||||
|
* **ci:** :zap: github action ([4142ca4](https://github.com/leifermendez/bot-whatsapp/commit/4142ca4fd552e7005f3b1397a76b90a2e574d19d))
|
||||||
|
* **ci:** :zap: github action ([091544a](https://github.com/leifermendez/bot-whatsapp/commit/091544ac3fac0c16925e856e1aec64bcad0ecf6d))
|
||||||
|
* **ci:** :zap: github action ([2ce342a](https://github.com/leifermendez/bot-whatsapp/commit/2ce342a0cb15019d5084ca06dc30e342b030ea10))
|
||||||
|
* **ci:** :zap: github action ([7817793](https://github.com/leifermendez/bot-whatsapp/commit/781779328f93ef8b0e6e0f85c6cd05ae782112fb))
|
||||||
|
* **ci:** :zap: github action ([dfced8c](https://github.com/leifermendez/bot-whatsapp/commit/dfced8c594e9175c81e837af359631ba055b7e1a))
|
||||||
|
* **ci:** :zap: github action ([aaa4ce8](https://github.com/leifermendez/bot-whatsapp/commit/aaa4ce837229fd51e274de3d91e1d9d615ac69fd))
|
||||||
|
* **ci:** :zap: github action ([9ddf144](https://github.com/leifermendez/bot-whatsapp/commit/9ddf144244cd6877e7d26f576387814459f2befb))
|
||||||
|
* **ci:** :zap: github action ([b465de5](https://github.com/leifermendez/bot-whatsapp/commit/b465de55a0e511213d1a7760a74efa102172c85e))
|
||||||
|
* **ci:** :zap: github action ([cf1dc6f](https://github.com/leifermendez/bot-whatsapp/commit/cf1dc6fac810545e5a2b63f31f71322f37329e38))
|
||||||
|
* **ci:** :zap: github action ([8d897f8](https://github.com/leifermendez/bot-whatsapp/commit/8d897f824e27a55ca011163092a813a7e8f426af))
|
||||||
|
* **ci:** ci ([f55cfae](https://github.com/leifermendez/bot-whatsapp/commit/f55cfae6e4ccc1df949212999406680020d27f9c))
|
||||||
|
* **ci:** ci ([671c5b3](https://github.com/leifermendez/bot-whatsapp/commit/671c5b37f33360e8cb754625b8dd6e83bce9014d))
|
||||||
|
* **cli:** :bug: path ([32212fb](https://github.com/leifermendez/bot-whatsapp/commit/32212fb52d206bf6f8d753a86d9ce40aa0db2a5d))
|
||||||
|
* **cli:** :fire: create script - templates ([2319db3](https://github.com/leifermendez/bot-whatsapp/commit/2319db3009501fe57ae21e60ad286eb68c46f4fd))
|
||||||
|
* **cli:** :fire: create script - templates ([9cb98b5](https://github.com/leifermendez/bot-whatsapp/commit/9cb98b5e73fca3c3f5e70a8497badc31e494b943))
|
||||||
|
* **cli:** :fire: create script - templates ([2999e0e](https://github.com/leifermendez/bot-whatsapp/commit/2999e0e753f31a8b9e6d7c117e78cdb5656e203a))
|
||||||
|
* **cli:** :fire: create script - templates ([af716b7](https://github.com/leifermendez/bot-whatsapp/commit/af716b75372899877a81b528b58278376166d0ad))
|
||||||
|
* **cli:** :fire: create script - templates ([c6999c8](https://github.com/leifermendez/bot-whatsapp/commit/c6999c84931083a87b5717db58003be68244707e))
|
||||||
|
* **cli:** :fire: create script - templates ([d4b49a9](https://github.com/leifermendez/bot-whatsapp/commit/d4b49a9bd7085070f0c5964d2903f10b71bde0b3))
|
||||||
|
* **cli:** :fire: create script - templates ([eebc3c9](https://github.com/leifermendez/bot-whatsapp/commit/eebc3c980638d88f11a0d93b8344f3ff345c7ee5))
|
||||||
|
* **cli:** :zap: clean eslinter ([bfb69d9](https://github.com/leifermendez/bot-whatsapp/commit/bfb69d9a9574a757ae02748b6c5f5afa3eac68e6))
|
||||||
|
* **cli:** :zap: clean eslinter ([15f6972](https://github.com/leifermendez/bot-whatsapp/commit/15f697225775a0f0e0a440cd980f7fb8f51a1056))
|
||||||
|
* **cli:** :zap: create-starter ([d3b8310](https://github.com/leifermendez/bot-whatsapp/commit/d3b8310180d2ad813733b1d18f2c32d7d947740a))
|
||||||
|
* **cli:** :zap: update cli copy ([7797c2b](https://github.com/leifermendez/bot-whatsapp/commit/7797c2b46133697e2a591adab2b67e66b34a1cfe))
|
||||||
|
* **fix:** fix ([6483545](https://github.com/leifermendez/bot-whatsapp/commit/648354500b123f20044f5ac2e8a26b15f16d1b8d))
|
||||||
|
* **fix:** fix ([28c0480](https://github.com/leifermendez/bot-whatsapp/commit/28c0480b8bfa6b24394095f57c36ef89c9aeb566))
|
||||||
|
* **linter:** update linter and commitlint ([70a94ab](https://github.com/leifermendez/bot-whatsapp/commit/70a94ab2c6f8e4122780c77bc3a621944883e621))
|
||||||
|
* pre-copy fix ([08e2552](https://github.com/leifermendez/bot-whatsapp/commit/08e2552907c48cfeaac843457a18bf2032e6f8aa))
|
||||||
|
* pre-copy fix ([6617107](https://github.com/leifermendez/bot-whatsapp/commit/6617107ab824215c449e26eae6c2bb327ecfc092))
|
||||||
|
* **starter:** pre-copy fix ([929e74c](https://github.com/leifermendez/bot-whatsapp/commit/929e74c84b667ec13cb5490b3b951cb8df15ebd1))
|
||||||
|
|
||||||
|
|
||||||
|
* (💍) Is justa test! ([37d04e9](https://github.com/leifermendez/bot-whatsapp/commit/37d04e9e89d3f01fdc367654ba60fb11ab2614c4))
|
||||||
|
|
||||||
|
### [0.0.3](https://github.com/leifermendez/bot-whatsapp/compare/v0.2.0...v0.0.3) (2022-12-12)
|
||||||
|
|
||||||
|
## 0.2.0 (2022-12-12)
|
||||||
|
|
||||||
|
|
||||||
|
### ⚠ BREAKING CHANGES
|
||||||
|
|
||||||
|
* 🧨 NO
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* (🎸) add onClick prop to component ([4ae3898](https://github.com/leifermendez/bot-whatsapp/commit/4ae389846d38c133f6bb2129ae373eed39d9d08d))
|
||||||
|
* **adapter:** added adapter mysql ([717a7dc](https://github.com/leifermendez/bot-whatsapp/commit/717a7dc95fbc107ec6f55387aff606c47144baa4))
|
||||||
|
* **adapter:** implementation of json file adapter ([5e1a373](https://github.com/leifermendez/bot-whatsapp/commit/5e1a3737303c843095984f6357564ea18458362f))
|
||||||
|
* **adapter:** mysql adapter ([8d73c86](https://github.com/leifermendez/bot-whatsapp/commit/8d73c86946d07aa80e5b375b62b84a88b2892e03))
|
||||||
|
* **adapter:** sql is added to create the table ([4b7de0f](https://github.com/leifermendez/bot-whatsapp/commit/4b7de0f6901524fa2c09271c3a99c364e6b3c260))
|
||||||
|
* **bot:** :fire: improvement provider handler ([4154cc2](https://github.com/leifermendez/bot-whatsapp/commit/4154cc223091a46d3203d3a378cd42f61749a5fa))
|
||||||
|
* **bot:** :zap: add send image function ([ce8a96b](https://github.com/leifermendez/bot-whatsapp/commit/ce8a96b958ff814c65d8fb4dbd5eaed5cc83a1ed))
|
||||||
|
* **ci:** :art: add releases ([b115dc3](https://github.com/leifermendez/bot-whatsapp/commit/b115dc3654996f049837bfb6b5d039a2313df0ad))
|
||||||
|
* **ci:** :art: relases script ([2e906bc](https://github.com/leifermendez/bot-whatsapp/commit/2e906bce79f7d854d437334e1d0c2cd270b0bbc6))
|
||||||
|
* **ci:** update ci ([e5a9db7](https://github.com/leifermendez/bot-whatsapp/commit/e5a9db7e12340c4f5baa66e8b20585b63daa3bcf))
|
||||||
|
* **cli:** create-starter ([3977987](https://github.com/leifermendez/bot-whatsapp/commit/397798790ef5857ca758b0df8384c6a4bfacc181))
|
||||||
|
* **provider:** :fire: add twilii (weoking) ([4350dff](https://github.com/leifermendez/bot-whatsapp/commit/4350dff22a7de69ba6d35ecbdd67e59b810bd46f))
|
||||||
|
* **provider:** added twilio provider ([8dd3be9](https://github.com/leifermendez/bot-whatsapp/commit/8dd3be909b36717f6b54e141a5f48d2722d4855c))
|
||||||
|
* **release:** added ([1988948](https://github.com/leifermendez/bot-whatsapp/commit/1988948c30d922beb7b83faab96d1d59cf7f5f90))
|
||||||
|
* **release:** added ([f4ad704](https://github.com/leifermendez/bot-whatsapp/commit/f4ad7040abf619635480c30babd6f1159c7af85a))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **adapter:** conflict resolution ([4b307ef](https://github.com/leifermendez/bot-whatsapp/commit/4b307efe79c738a5c4e04ff1c07ca247d827593c))
|
||||||
|
* **adapter:** corrections are made to the adapter ([afa6771](https://github.com/leifermendez/bot-whatsapp/commit/afa677190392d48715930ebe2b1e15c7619d730f))
|
||||||
|
* **bot:** :zap: added delay promises ([73caf09](https://github.com/leifermendez/bot-whatsapp/commit/73caf090ba9013132e5dcb7761a10939dc9ac300))
|
||||||
|
* **bot:** :zap: fix sensitive case ([24ac9fb](https://github.com/leifermendez/bot-whatsapp/commit/24ac9fbf48f80eeb521a36bc938af3a70dd82303))
|
||||||
|
* **bot:** :zap: flow improvement + add utils ([a7b19d9](https://github.com/leifermendez/bot-whatsapp/commit/a7b19d9bff5ea66ff888555c3df37ae0e20b612a))
|
||||||
|
* **bot:** update ([49698bf](https://github.com/leifermendez/bot-whatsapp/commit/49698bfda9d2a53f7b1a7e1724a796698601fbaa))
|
||||||
|
* **ci:** :zap: balance version ([ec46cfd](https://github.com/leifermendez/bot-whatsapp/commit/ec46cfdd657c08c8e90261613f00cfc080f1e1d6))
|
||||||
|
* **ci:** :zap: github action ([b827a0a](https://github.com/leifermendez/bot-whatsapp/commit/b827a0ab225b89bb8117c82628db0679c09b4102))
|
||||||
|
* **ci:** :zap: github action ([4142ca4](https://github.com/leifermendez/bot-whatsapp/commit/4142ca4fd552e7005f3b1397a76b90a2e574d19d))
|
||||||
|
* **ci:** :zap: github action ([091544a](https://github.com/leifermendez/bot-whatsapp/commit/091544ac3fac0c16925e856e1aec64bcad0ecf6d))
|
||||||
|
* **ci:** :zap: github action ([2ce342a](https://github.com/leifermendez/bot-whatsapp/commit/2ce342a0cb15019d5084ca06dc30e342b030ea10))
|
||||||
|
* **ci:** :zap: github action ([7817793](https://github.com/leifermendez/bot-whatsapp/commit/781779328f93ef8b0e6e0f85c6cd05ae782112fb))
|
||||||
|
* **ci:** :zap: github action ([dfced8c](https://github.com/leifermendez/bot-whatsapp/commit/dfced8c594e9175c81e837af359631ba055b7e1a))
|
||||||
|
* **ci:** :zap: github action ([aaa4ce8](https://github.com/leifermendez/bot-whatsapp/commit/aaa4ce837229fd51e274de3d91e1d9d615ac69fd))
|
||||||
|
* **ci:** :zap: github action ([9ddf144](https://github.com/leifermendez/bot-whatsapp/commit/9ddf144244cd6877e7d26f576387814459f2befb))
|
||||||
|
* **ci:** :zap: github action ([b465de5](https://github.com/leifermendez/bot-whatsapp/commit/b465de55a0e511213d1a7760a74efa102172c85e))
|
||||||
|
* **ci:** :zap: github action ([cf1dc6f](https://github.com/leifermendez/bot-whatsapp/commit/cf1dc6fac810545e5a2b63f31f71322f37329e38))
|
||||||
|
* **ci:** :zap: github action ([8d897f8](https://github.com/leifermendez/bot-whatsapp/commit/8d897f824e27a55ca011163092a813a7e8f426af))
|
||||||
|
* **ci:** ci ([f55cfae](https://github.com/leifermendez/bot-whatsapp/commit/f55cfae6e4ccc1df949212999406680020d27f9c))
|
||||||
|
* **ci:** ci ([671c5b3](https://github.com/leifermendez/bot-whatsapp/commit/671c5b37f33360e8cb754625b8dd6e83bce9014d))
|
||||||
|
* **cli:** :bug: path ([32212fb](https://github.com/leifermendez/bot-whatsapp/commit/32212fb52d206bf6f8d753a86d9ce40aa0db2a5d))
|
||||||
|
* **cli:** :fire: create script - templates ([2319db3](https://github.com/leifermendez/bot-whatsapp/commit/2319db3009501fe57ae21e60ad286eb68c46f4fd))
|
||||||
|
* **cli:** :fire: create script - templates ([9cb98b5](https://github.com/leifermendez/bot-whatsapp/commit/9cb98b5e73fca3c3f5e70a8497badc31e494b943))
|
||||||
|
* **cli:** :fire: create script - templates ([2999e0e](https://github.com/leifermendez/bot-whatsapp/commit/2999e0e753f31a8b9e6d7c117e78cdb5656e203a))
|
||||||
|
* **cli:** :fire: create script - templates ([af716b7](https://github.com/leifermendez/bot-whatsapp/commit/af716b75372899877a81b528b58278376166d0ad))
|
||||||
|
* **cli:** :fire: create script - templates ([c6999c8](https://github.com/leifermendez/bot-whatsapp/commit/c6999c84931083a87b5717db58003be68244707e))
|
||||||
|
* **cli:** :fire: create script - templates ([d4b49a9](https://github.com/leifermendez/bot-whatsapp/commit/d4b49a9bd7085070f0c5964d2903f10b71bde0b3))
|
||||||
|
* **cli:** :fire: create script - templates ([eebc3c9](https://github.com/leifermendez/bot-whatsapp/commit/eebc3c980638d88f11a0d93b8344f3ff345c7ee5))
|
||||||
|
* **cli:** :zap: clean eslinter ([bfb69d9](https://github.com/leifermendez/bot-whatsapp/commit/bfb69d9a9574a757ae02748b6c5f5afa3eac68e6))
|
||||||
|
* **cli:** :zap: clean eslinter ([15f6972](https://github.com/leifermendez/bot-whatsapp/commit/15f697225775a0f0e0a440cd980f7fb8f51a1056))
|
||||||
|
* **cli:** :zap: create-starter ([d3b8310](https://github.com/leifermendez/bot-whatsapp/commit/d3b8310180d2ad813733b1d18f2c32d7d947740a))
|
||||||
|
* **cli:** :zap: update cli copy ([7797c2b](https://github.com/leifermendez/bot-whatsapp/commit/7797c2b46133697e2a591adab2b67e66b34a1cfe))
|
||||||
|
* **fix:** fix ([6483545](https://github.com/leifermendez/bot-whatsapp/commit/648354500b123f20044f5ac2e8a26b15f16d1b8d))
|
||||||
|
* **fix:** fix ([28c0480](https://github.com/leifermendez/bot-whatsapp/commit/28c0480b8bfa6b24394095f57c36ef89c9aeb566))
|
||||||
|
* **linter:** update linter and commitlint ([70a94ab](https://github.com/leifermendez/bot-whatsapp/commit/70a94ab2c6f8e4122780c77bc3a621944883e621))
|
||||||
|
* pre-copy fix ([08e2552](https://github.com/leifermendez/bot-whatsapp/commit/08e2552907c48cfeaac843457a18bf2032e6f8aa))
|
||||||
|
* pre-copy fix ([6617107](https://github.com/leifermendez/bot-whatsapp/commit/6617107ab824215c449e26eae6c2bb327ecfc092))
|
||||||
|
* **starter:** pre-copy fix ([929e74c](https://github.com/leifermendez/bot-whatsapp/commit/929e74c84b667ec13cb5490b3b951cb8df15ebd1))
|
||||||
|
|
||||||
|
|
||||||
|
* (💍) Is justa test! ([37d04e9](https://github.com/leifermendez/bot-whatsapp/commit/37d04e9e89d3f01fdc367654ba60fb11ab2614c4))
|
||||||
|
|
||||||
|
## 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
|
||||||
- remove ora and chalk
|
- remove ora and chalk
|
||||||
- add env
|
- add env
|
||||||
|
|||||||
103
CONTRIBUTING.md
Normal file
103
CONTRIBUTING.md
Normal file
@@ -0,0 +1,103 @@
|
|||||||
|
# CONTRIBUTING
|
||||||
|
|
||||||
|
### 📄 Bienvenido/a
|
||||||
|
Si deseas colaborar con el proyecto existen varias maneras, la primera de ellas es aportando conocimiento y mejorando el repositorio (actualizando documentación, mejorando código, revisando __[issues](https://github.com/codigoencasa/bot-whatsapp/issues)__, etc).
|
||||||
|
|
||||||
|
También es bien recibido los aportes económicos que se utilizaran para diferentes fines __[ver más](https://opencollective.com/bot-whatsapp)__
|
||||||
|
|
||||||
|
El lenguaje principal que se utilizó para desarrollar este proyecto fue __JavaScript__ con el fin de qué personas que están iniciando en el mundo de la programación puedan entender fácilmente.
|
||||||
|
|
||||||
|
|
||||||
|
### 🤔 Preguntas frecuentes
|
||||||
|
- ¿Como puedo hacer aportaciones de código en el proyecto?: [Ver Video](https://youtu.be/Lxt8Acob6aU)
|
||||||
|
- ¿Como ejecutar el entorno de pruebas?: [Ver Video](https://youtu.be/Mf9V-dloBfk)
|
||||||
|
- ¿Como crear un nuevo proveedor?: [Ver Video](https://youtu.be/cahK9zH3SI8)
|
||||||
|
- ¿Que son los GithubActions?: [Ver Video](https://youtu.be/nYBEBFKLiqw)
|
||||||
|
- ¿Canales de comunicación?: [Discord](https://link.codigoencasa.com/DISCORD)
|
||||||
|
|
||||||
|
-----
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
__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 conseguirás las intrucciones para instalar yarn.
|
||||||
|
- __[VSCode](https://code.visualstudio.com/download)__ (recomendado): Editor de código con plugins.
|
||||||
|
- __[Conventional Commits](https://marketplace.visualstudio.com/items?itemName=vivaxy.vscode-conventional-commits&ssr=false#overview)__ (plugin-vscode) este plugin te ayudará a crear commit semántico.
|
||||||
|
- Se usará 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/codigoencasa/bot-whatsapp
|
||||||
|
```
|
||||||
|
__Instalar dependencias__
|
||||||
|
```
|
||||||
|
cd bot-whatsapp
|
||||||
|
yarn install
|
||||||
|
```
|
||||||
|
|
||||||
|
__Compilar (build)__
|
||||||
|
Para compilar la aplicación es necesario ejecutar este comando, el cual genera un directorio `lib` dentro de los paquetes del monorepo.
|
||||||
|
|
||||||
|
```
|
||||||
|
yarn build
|
||||||
|
```
|
||||||
|
|
||||||
|
__Example-app__
|
||||||
|
Se ejecuta el CLI (Command Line Interface) para ayudarte a crear un app-bot de ejemplo.
|
||||||
|
```
|
||||||
|
yarn run cli
|
||||||
|
```
|
||||||
|
|
||||||
|
Selecionas (mediante las flechas arriba y abajo) el proveedor que quieras usar y cuando estes sobre el presiona la barra de espacio, igualmente selecciona la base de datos que quieras usar.
|
||||||
|
|
||||||
|
Se creó un subdirecorio con el nombre del proveedor y base de datos que seleccionaste, ejemplo: `base-bailey-mysql`
|
||||||
|
|
||||||
|
Dentro de ese directorio necesitas editar el archivo package.json y borrar las siguientes lineas:
|
||||||
|
```
|
||||||
|
"@bot-whatsapp/bot": "latest",
|
||||||
|
"@bot-whatsapp/cli": "latest",
|
||||||
|
"@bot-whatsapp/database": "latest",
|
||||||
|
"@bot-whatsapp/provider": "latest",
|
||||||
|
```
|
||||||
|
|
||||||
|
Cambiate al directorio creado ejemplo: `base-bailey-mysql`
|
||||||
|
```
|
||||||
|
cd base-baileys-mysql
|
||||||
|
```
|
||||||
|
Ejecuta los comandos:
|
||||||
|
```
|
||||||
|
npm install
|
||||||
|
npm run pre-copy
|
||||||
|
npm start
|
||||||
|
```
|
||||||
|
En el caso de MySql y Mongo es necesario especificar en app.js los datos de la conexión, ejemplo de MySql:
|
||||||
|
```
|
||||||
|
const BaileysProvider = require('@bot-whatsapp/provider/baileys')
|
||||||
|
const MySQLAdapter = require('@bot-whatsapp/database/mysql')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Declaramos las conexiones de MySQL
|
||||||
|
*/
|
||||||
|
const MYSQL_DB_HOST = 'localhost'
|
||||||
|
const MYSQL_DB_USER = 'usr'
|
||||||
|
const MYSQL_DB_PASSWORD = 'pass'
|
||||||
|
const MYSQL_DB_NAME = 'bot'
|
||||||
|
```
|
||||||
|
<!-- __Seguir instrucciones__
|
||||||
|
En la consola encontraras los pasos a seguir -->
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
> __NOTA:__ [Eres libre de aportar informacion a este documento o arreglar ortografia 🤣](
|
||||||
|
https://github.com/codigoencasa/bot-whatsapp/edit/dev/CONTRIBUTING.md)
|
||||||
|
|
||||||
|
------
|
||||||
|
- [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)
|
||||||
88
EXAMPLE.md
Normal file
88
EXAMPLE.md
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
|
||||||
|
```js
|
||||||
|
|
||||||
|
const {
|
||||||
|
createBot,
|
||||||
|
createProvider,
|
||||||
|
createFlow,
|
||||||
|
addKeyword,
|
||||||
|
toSerialize,
|
||||||
|
} = require('@bot-whatsapp/bot')
|
||||||
|
|
||||||
|
const WebWhatsappProvider = require('@bot-whatsapp/provider/web-whatsapp')
|
||||||
|
const MongoAdapter = require('@bot-whatsapp/database/mongo')
|
||||||
|
|
||||||
|
const flowArepa1 = toSerialize(
|
||||||
|
addKeyword(['1', 'AREPA14'])
|
||||||
|
.addAnswer('Esta es una arepa calificada ⭐⭐⭐⭐⭐')
|
||||||
|
.addAnswer(['Ingredientes:', '10g Aguacate', '20g Huevo'].join('\n'))
|
||||||
|
.toJson()
|
||||||
|
)
|
||||||
|
|
||||||
|
const flowArepa2_2 = toSerialize(
|
||||||
|
addKeyword('SI').addAnswer('te pongo huevo de mentira!').toJson()
|
||||||
|
)
|
||||||
|
|
||||||
|
const flowArepa2 = toSerialize(
|
||||||
|
addKeyword(['arepa2'])
|
||||||
|
.addAnswer('Esta es una arepa calificada ⭐⭐⭐⭐')
|
||||||
|
.addAnswer(
|
||||||
|
['Ingredientes:', '10g perico', '20g huevo', '10g queso'].join('\n')
|
||||||
|
)
|
||||||
|
.addAnswer(
|
||||||
|
'Eres Vegano SI o NO',
|
||||||
|
{
|
||||||
|
capture: true,
|
||||||
|
},
|
||||||
|
null,
|
||||||
|
[...flowArepa2_2]
|
||||||
|
)
|
||||||
|
.toJson()
|
||||||
|
)
|
||||||
|
|
||||||
|
const flowArepa3 = toSerialize(
|
||||||
|
addKeyword(['arepa3'])
|
||||||
|
.addAnswer('Esta es una arepa calificada LAMEJOR ⭐⭐⭐⭐⭐')
|
||||||
|
.toJson()
|
||||||
|
)
|
||||||
|
|
||||||
|
//////////////--MENU--PRINCIPAL--//////////////////
|
||||||
|
|
||||||
|
const flujoMenuArepa = addKeyword(['hola', 'ola', 'buenos'])
|
||||||
|
.addAnswer('Bienvenido "Arepera Aji Picante 🤯🚀😅"')
|
||||||
|
.addAnswer(
|
||||||
|
[
|
||||||
|
'El menú de hoy es el siguiente:',
|
||||||
|
'👉 [1 -AREPA14] - Arepa tradicional con Aguacate y Huevo',
|
||||||
|
'👉 [arepa2] - Arepa rellena de perico y huevo con un toque de queso',
|
||||||
|
'👉 [arepa3] - Rellena de Jamon y Queso',
|
||||||
|
].join('\n')
|
||||||
|
)
|
||||||
|
.addAnswer(
|
||||||
|
'Esperando respuesta...',
|
||||||
|
{
|
||||||
|
capture: true,
|
||||||
|
},
|
||||||
|
() => {
|
||||||
|
console.log('Enviar un mail!')
|
||||||
|
},
|
||||||
|
[...flowArepa1, ...flowArepa2, ...flowArepa3]
|
||||||
|
)
|
||||||
|
.addAnswer('Gracias!')
|
||||||
|
|
||||||
|
const main = async () => {
|
||||||
|
const adapterDB = new MongoAdapter()
|
||||||
|
const adapterFlow = createFlow([flujoMenuArepa])
|
||||||
|
const adapterProvider = createProvider(WebWhatsappProvider)
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
flow: adapterFlow,
|
||||||
|
provider: adapterProvider,
|
||||||
|
database: adapterDB,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
main()
|
||||||
|
|
||||||
|
|
||||||
|
```
|
||||||
2
GLOSSARY.md
Normal file
2
GLOSSARY.md
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
CTX: Es el objeto que representa un mensaje, con opciones, id, ref
|
||||||
|
messageInComming: Objeto entrante del provider {body, from,to,...}
|
||||||
214
MIGRATION.md
Normal file
214
MIGRATION.md
Normal file
@@ -0,0 +1,214 @@
|
|||||||
|
# Migración
|
||||||
|
|
||||||
|
#### Versión (legacy)
|
||||||
|
|
||||||
|
En la ***versión (legacy)*** se implementaban los flujos de esta manera, en dos archivos independientes.
|
||||||
|
|
||||||
|
> __`initial.json`__ para establecer las palabras claves y el flujo a responder, por otro lado tambien se necesitaba implementar
|
||||||
|
> __`response.json`__ donde se escriben los mensajes a responder.
|
||||||
|
|
||||||
|
```json
|
||||||
|
//initial.json
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"keywords": [
|
||||||
|
"hola",
|
||||||
|
"ola",
|
||||||
|
"alo"
|
||||||
|
],
|
||||||
|
"key": "hola"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"keywords": ["productos", "info"],
|
||||||
|
"key": "productos"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"keywords": ["adios", "bye"],
|
||||||
|
"key": "adios"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"keywords": ["imagen", "foto"],
|
||||||
|
"key": "catalogo"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
```
|
||||||
|
```json
|
||||||
|
//response.json
|
||||||
|
{
|
||||||
|
"hola":{
|
||||||
|
"replyMessage":[
|
||||||
|
"Gracias a ti! \n"
|
||||||
|
],
|
||||||
|
"media":null,
|
||||||
|
"trigger":null
|
||||||
|
},
|
||||||
|
"adios":{
|
||||||
|
"replyMessage":[
|
||||||
|
"Que te vaya bien!!"
|
||||||
|
],
|
||||||
|
},
|
||||||
|
"productos":{
|
||||||
|
"replyMessage":[
|
||||||
|
"Más productos aquí"
|
||||||
|
],
|
||||||
|
"trigger":null,
|
||||||
|
"actions":{
|
||||||
|
"title":"¿Que te interesa ver?",
|
||||||
|
"message":"Abajo unos botons",
|
||||||
|
"footer":"",
|
||||||
|
"buttons":[
|
||||||
|
{"body":"Telefonos"},
|
||||||
|
{"body":"Computadoras"},
|
||||||
|
{"body":"Otros"}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"catalogo":{
|
||||||
|
"replyMessage":[
|
||||||
|
"Te envio una imagen"
|
||||||
|
],
|
||||||
|
"media":"https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif",
|
||||||
|
"trigger":null,
|
||||||
|
},
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Versión 2 (0.2.X)
|
||||||
|
|
||||||
|
En esta versión es mucho más sencillo, abajo encontraras un ejemplo del mismo flujo anteriormente mostrado.
|
||||||
|
|
||||||
|
```js
|
||||||
|
//app.js
|
||||||
|
const {
|
||||||
|
createBot,
|
||||||
|
createProvider,
|
||||||
|
createFlow,
|
||||||
|
addKeyword,
|
||||||
|
addChild,
|
||||||
|
} = require('@bot-whatsapp/bot')
|
||||||
|
|
||||||
|
const BaileysProvider = require('@bot-whatsapp/provider/baileys')
|
||||||
|
const MockAdapter = require('@bot-whatsapp/database/mock')
|
||||||
|
/**
|
||||||
|
* Declarando flujos principales.
|
||||||
|
*/
|
||||||
|
const flowHola = addKeyword(['hola', 'ola', 'alo'])
|
||||||
|
.addAnswer('Bienvenido a tu tienda online!')
|
||||||
|
|
||||||
|
const flowAdios = addKeyword(['adios', 'bye'])
|
||||||
|
.addAnswer('Que te vaya bien!!')
|
||||||
|
.addAnswer('Hasta luego!')
|
||||||
|
|
||||||
|
const flowProductos = addKeyword(['productos', 'info'])
|
||||||
|
.addAnswer('Te envio una imagen', {
|
||||||
|
buttons:[
|
||||||
|
{body:"Telefonos"},
|
||||||
|
{body:"Computadoras"},
|
||||||
|
{body:"Otros"}
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
|
const flowCatalogo = addKeyword(['imagen', 'foto'])
|
||||||
|
.addAnswer('Te envio una imagen', {media:'https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif'})
|
||||||
|
|
||||||
|
|
||||||
|
const main = async () => {
|
||||||
|
const adapterDB = new MockAdapter()
|
||||||
|
const adapterFlow = createFlow([flowHola, flowAdios, flowProductos, flowCatalogo])
|
||||||
|
const adapterProvider = createProvider(BaileysProvider)
|
||||||
|
createBot({
|
||||||
|
flow: adapterFlow,
|
||||||
|
provider: adapterProvider,
|
||||||
|
database: adapterDB,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Flujos hijos
|
||||||
|
|
||||||
|
A continuación se muestra un ejemplo de flujos hijos, estos nos sirven para crear flujos que solo se disparan cuando el flujo anterior es el especificado, ejemplo:
|
||||||
|
|
||||||
|
> Menu Principal (Escoge zapatos o bolsos)
|
||||||
|
> - SubMenu 1 (Elegiste bolsos, ahora escoge piel o tela)
|
||||||
|
> - Submenu 1.1 (piel)
|
||||||
|
> - Submenu 2 (Elegiste zapatos, ahora escoge piel o tela)
|
||||||
|
> - Submenu 2.1 (piel)
|
||||||
|
|
||||||
|
El __submenu 1__ solo se va a disparar cuando el flujo anterior sea el __principal__, e igualmente el __submenu 1.1__, solo cuando el flujo anterior sea el __submenu 1__, ejemplo:
|
||||||
|
|
||||||
|
```js
|
||||||
|
/**
|
||||||
|
* Aqui declaramos los flujos hijos, los flujos se declaran de atras para adelante, es decir que si tienes un flujo de este tipo:
|
||||||
|
*
|
||||||
|
* Menu Principal
|
||||||
|
* - SubMenu 1
|
||||||
|
* - Submenu 1.1
|
||||||
|
* - Submenu 2
|
||||||
|
* - Submenu 2.1
|
||||||
|
*
|
||||||
|
* Primero declaras los submenus 1.1 y 2.1, luego el 1 y 2 y al final el principal.
|
||||||
|
*/
|
||||||
|
const flowBolsos2 = addKeyword(['bolsos2', '2'])
|
||||||
|
.addAnswer('🤯 *MUCHOS* bolsos ...')
|
||||||
|
.addAnswer('y mas bolsos... bla bla')
|
||||||
|
|
||||||
|
const flowZapatos2 = addKeyword(['zapatos2', '2'])
|
||||||
|
.addAnswer('🤯 repito que tengo *MUCHOS* zapatos.')
|
||||||
|
.addAnswer('y algunas otras cosas.')
|
||||||
|
|
||||||
|
const flowZapatos = addKeyword(['1', 'zapatos', 'ZAPATOS'])
|
||||||
|
.addAnswer('🤯 Veo que elegiste zapatos')
|
||||||
|
.addAnswer('Tengo muchos zapatos...bla bla')
|
||||||
|
.addAnswer(
|
||||||
|
['Manda:', '*(2) Zapatos2*', 'para mas información'],
|
||||||
|
{ 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(flowZapatos2)]
|
||||||
|
)
|
||||||
|
|
||||||
|
const flowBolsos = addKeyword(['2', 'bolsos', 'BOLSOS'])
|
||||||
|
.addAnswer('🙌 Veo que elegiste bolsos')
|
||||||
|
.addAnswer('Tengo muchos bolsos...bla bla')
|
||||||
|
.addAnswer(
|
||||||
|
['Manda:', '*(2) Bolsos2*', 'para mas información.'],
|
||||||
|
{ 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(flowBolsos2)]
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Declarando flujo principal
|
||||||
|
*/
|
||||||
|
|
||||||
|
const flowPrincipal = addKeyword(['hola', 'ole', 'alo'])
|
||||||
|
.addAnswer(['Hola, bienvenido a mi tienda', '¿Como puedo ayudarte?'])
|
||||||
|
.addAnswer(['Tengo:', 'Zapatos', 'Bolsos', 'etc ...'])
|
||||||
|
.addAnswer(
|
||||||
|
['Para continuar escribe:', '*(1) Zapatos*', '*(2) Bolsos*'],
|
||||||
|
{ 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)]
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
> Forma parte de este proyecto.
|
||||||
|
|
||||||
|
- [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)
|
||||||
265
README.md
265
README.md
@@ -1,116 +1,175 @@
|
|||||||
## Chatbot Whatsapp (OpenSource)
|
# Chatbot Library
|
||||||
|

|
||||||
|
[](http://commitizen.github.io/cz-cli/)
|
||||||
|
[](https://link.codigoencasa.com/DISCORD)
|
||||||
|
|
||||||
#### Actualización
|
<p align="center">
|
||||||
|
<img width="300" src="https://i.imgur.com/Oauef6t.png">
|
||||||
|
</p>
|
||||||
|
|
||||||
| Feature | Status |
|
|
||||||
| ------------- | ------------- |
|
|
||||||
| Dialogflow | ✅ |
|
|
||||||
| MySQL | ✅ |
|
|
||||||
| JSON File | ✅ |
|
|
||||||
| QR Scan (route) | ✅ |
|
|
||||||
| Easy deploy heroku | ✅ |
|
|
||||||
| Buttons | ✅ℹ️ (No funciona en multi-device)|
|
|
||||||
| Send Voice Note | ✅ |
|
|
||||||
| Add support ubuntu/linux | ✅ |
|
|
||||||
|
|
||||||
## Requisitos
|
**Con esta librería, puedes construir flujos automatizados de conversación de manera agnóstica al proveedor de WhatsApp,** configurar respuestas automatizadas para preguntas frecuentes, recibir y responder mensajes de manera automatizada, y hacer un seguimiento de las interacciones con los clientes. Además, puedes configurar fácilmente disparadores que te ayudaran a expandir las funcionalidades sin límites. **[Ver más informacion](https://bot-whatsapp.netlify.app/)**
|
||||||
- node v14 o superior
|
|
||||||
- VSCode (Editor de codigo) [Descargar](https://code.visualstudio.com/download)
|
|
||||||
- MySql (opcional) solo aplica si vas a usar el modo 'mysql' [sql-bot.sql migración](https://github.com/leifermendez/bot-whatsapp/blob/main/sql-bot.sql)
|
|
||||||
- Dialogflow (opcional) solo aplica si vas a usar el modo 'dialogflow'
|
|
||||||
|
|
||||||
### (Nuevo) Botones
|
## Comenzar
|
||||||
|
|
||||||
[](https://youtu.be/5lEMCeWEJ8o)
|
|
||||||
|
|
||||||
> Implementar los botones solo necesitas hacer uso del metodo __sendMessageButton__ que se encuentra dentro `./controllers/send` dejo un ejemplo de como usarlo.
|
|
||||||
[Ver implementación](https://github.com/leifermendez/bot-whatsapp/blob/main/app.js#L123)
|
|
||||||
|
|
||||||
``` javascript
|
|
||||||
const { sendMessageButton } = require('./controllers/send')
|
|
||||||
|
|
||||||
await sendMessageButton(
|
|
||||||
{
|
|
||||||
"title":"¿Que te interesa ver?",
|
|
||||||
"message":"Recuerda todo este contenido es gratis y estaria genial que me siguas!",
|
|
||||||
"footer":"Gracias",
|
|
||||||
"buttons":[
|
|
||||||
{"body":"😎 Cursos"},
|
|
||||||
{"body":"👉 Youtube"},
|
|
||||||
{"body":"😁 Telegram"}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
```
|
```
|
||||||
|
npm create bot-whatsapp@latest
|
||||||
## Notas de Voz
|
|
||||||
[](https://i.imgur.com/zq6xYDp.png)
|
|
||||||
|
|
||||||
> Se pueden enviar notas de voz con formato nativo para que no se vea como reenviado. En este ejemplo enviare el archivo __PTT-20220223-WA0000.opus__ que se encuentra dentro de la carpeta de __/mediaSend__
|
|
||||||
|
|
||||||
``` javascript
|
|
||||||
const { sendMediaVoiceNote } = require('./controllers/send')
|
|
||||||
|
|
||||||
await sendMediaVoiceNote(client, from, 'PTT-20220223-WA0000.opus')
|
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## Instruciones
|
Entiende más a fondo sus funcionalidades explicadas en nuestra documentación.
|
||||||
__Descargar o Clonar repositorio__
|
|
||||||
|
|
||||||
__Usas ¿Ubuntu / Linux?__
|
- Instalacion
|
||||||
> Asegurate de instalar los siguientes paquetes
|
- Base de datos
|
||||||
```
|
- Proveedores
|
||||||
sudo apt-get install -y libgbm-dev
|
|
||||||
sudo apt install -y gconf-service libasound2 libatk1.0-0 libc6 libcairo2 libcups2 libdbus-1-3 libexpat1 libfontconfig1 libgcc1 libgconf-2-4 libgdk-pixbuf2.0-0 libglib2.0-0 libgtk-3-0 libnspr4 libpango-1.0-0 libpangocairo-1.0-0 libstdc++6 libx11-6 libx11-xcb1 libxcb1 libxcomposite1 libxcursor1 libxdamage1 libxext6 libxfixes3 libxi6 libxrandr2 libxrender1 libxss1 libxtst6 ca-certificates fonts-liberation libappindicator1 libnss3 lsb-release xdg-utils wget
|
|
||||||
```
|
|
||||||
|
|
||||||
__Instalar dependencias (npm install)__
|
## Recursos
|
||||||
> Ubicate en le directorio que descargaste y via consola o terminal ejecuta el siguiente comando
|
- [📄 Documentación](https://bot-whatsapp.netlify.app/)
|
||||||
|
- [🚀 Roadmap](https://github.com/orgs/codigoencasa/projects/1)
|
||||||
|
- [💻 Discord](https://link.codigoencasa.com/DISCORD)
|
||||||
|
- [👌 Twitter](https://twitter.com/leifermendez)
|
||||||
|
- [🎥 Youtube](https://www.youtube.com/watch?v=5lEMCeWEJ8o&list=PL_WGMLcL4jzWPhdhcUyhbFU6bC0oJd2BR)
|
||||||
|
|
||||||
```
|
## Comunidad
|
||||||
npm i
|
<!-- readme: collaborators,contributors -start -->
|
||||||
```
|
<table>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/cheveguerra">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/5891114?v=4" width="50;" alt="cheveguerra"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Jose Alberto Guerra Ugalde</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/leifermendez">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/15802366?v=4" width="50;" alt="leifermendez"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Leifer Mendez</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/aurik3">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/37228512?v=4" width="50;" alt="aurik3"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Null</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/vicente1992">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/57806030?v=4" width="50;" alt="vicente1992"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Manuel Vicente Ortiz</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/leifermendezfroged">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/97020486?v=4" width="50;" alt="leifermendezfroged"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Leifer Mendez</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/danielcasta0398">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/98791147?v=4" width="50;" alt="danielcasta0398"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Juan Daniel Castaño</b></sub>
|
||||||
|
</a>
|
||||||
|
</td></tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/marianarolfo">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/68322254?v=4" width="50;" alt="marianarolfo"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Null</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/HKong31">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/113340082?v=4" width="50;" alt="HKong31"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>HLKong</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/jzvi12">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/10729787?v=4" width="50;" alt="jzvi12"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Zvi</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/JosephVTX">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/91026290?v=4" width="50;" alt="JosephVTX"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Joseph Vega Callupe</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/Gonzalito87">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/100331586?v=4" width="50;" alt="Gonzalito87"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Null</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/devrlbusiness">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/66280283?v=4" width="50;" alt="devrlbusiness"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Developer RL Business</b></sub>
|
||||||
|
</a>
|
||||||
|
</td></tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/Gregoriotecnico">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/118696506?v=4" width="50;" alt="Gregoriotecnico"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Null</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/jlferrete">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/36698913?v=4" width="50;" alt="jlferrete"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Jose Luis Ferrete Olarte</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/lisandroprada">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/7232326?v=4" width="50;" alt="lisandroprada"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Null</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/6rak0">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/12260031?v=4" width="50;" alt="6rak0"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Null</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/tonyvazgar">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/21047090?v=4" width="50;" alt="tonyvazgar"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Luis Antonio Vázquez García</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/rrruuuyyy">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/33061671?v=4" width="50;" alt="rrruuuyyy"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Rodrigo Mendoza Cabrera</b></sub>
|
||||||
|
</a>
|
||||||
|
</td></tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/yond1994">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/47557263?v=4" width="50;" alt="yond1994"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Yonathan Suarez</b></sub>
|
||||||
|
</a>
|
||||||
|
</td></tr>
|
||||||
|
</table>
|
||||||
|
<!-- readme: collaborators,contributors -end -->
|
||||||
|
|
||||||
__Configurar .env__
|
|
||||||
> Con el editor de texto crea un archivo `.env` el cual debes de guiarte del archivo `.env.example`
|
|
||||||
[Ver video explicando](https://youtu.be/5lEMCeWEJ8o?t=381)
|
|
||||||
```
|
|
||||||
######DATABASE: none, mysql, dialogflow
|
|
||||||
|
|
||||||
DEFAULT_MESSAGE=true
|
|
||||||
SAVE_MEDIA=true
|
|
||||||
PORT=3000
|
|
||||||
DATABASE=none
|
|
||||||
LANGUAGE=es
|
|
||||||
SQL_HOST=
|
|
||||||
SQL_USER=
|
|
||||||
SQL_PASS=
|
|
||||||
SQL_DATABASE=
|
|
||||||
```
|
|
||||||
|
|
||||||
__Ejecutar el script__
|
|
||||||
> Ubicate en le directorio que descargaste y via consola o terminal ejecuta el siguiente comando
|
|
||||||
`npm run start`
|
|
||||||
|
|
||||||
__Whatsapp en tu celular__
|
|
||||||
> Ahora abre la aplicación de Whatsapp en tu dispositivo y escanea el código QR
|
|
||||||
<img src="https://i.imgur.com/RSbPtat.png" width="500" />
|
|
||||||
Tambien puedes visitar la pagina http://127.0.0.1:3000/qr
|
|
||||||
|
|
||||||
__Listo 😎__
|
|
||||||
> Cuando sale este mensaje tu BOT está __listo__ para trabajar!
|
|
||||||

|
|
||||||
|
|
||||||
# ¿Quieres ver como se creó? 🤖
|
|
||||||
- [Ver Video 1](https://www.youtube.com/watch?v=A_Xu0OR_HkE)
|
|
||||||
- [¿Como instalarlo? (Actulización)](https://youtu.be/5lEMCeWEJ8o)
|
|
||||||
|
|
||||||
## ¿Como usarlo el chatbot de whatsapp?
|
|
||||||
> Escribe un mensaje al whatsapp que vinculaste con tu BOT
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
> Ahora deberías obtener un arespuesta por parte del BOT como la siguiente, ademas de esto tambien se crea un archivo excel
|
|
||||||
con el historial de conversación con el número de tu cliente
|
|
||||||
|
|||||||
21
SECURITY.md
Normal file
21
SECURITY.md
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# Security Policy
|
||||||
|
|
||||||
|
## Supported Versions
|
||||||
|
|
||||||
|
Use this section to tell people about which versions of your project are
|
||||||
|
currently being supported with security updates.
|
||||||
|
|
||||||
|
| Version | Supported |
|
||||||
|
| ------- | ------------------ |
|
||||||
|
| 5.1.x | :white_check_mark: |
|
||||||
|
| 5.0.x | :x: |
|
||||||
|
| 4.0.x | :white_check_mark: |
|
||||||
|
| < 4.0 | :x: |
|
||||||
|
|
||||||
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
|
Use this section to tell people how to report a vulnerability.
|
||||||
|
|
||||||
|
Tell them where to go, how often they can expect to get an update on a
|
||||||
|
reported vulnerability, what to expect if the vulnerability is accepted or
|
||||||
|
declined, etc.
|
||||||
51
TODO.md
Normal file
51
TODO.md
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
### Genral
|
||||||
|
- [X] __(doc)__ Video de como colaborar PR
|
||||||
|
- [ ] __(doc)__ Video implementación de test y cobertura
|
||||||
|
- [ ] __(doc)__ Video explicacion de github action
|
||||||
|
- [ ] Crear packages list externas
|
||||||
|
|
||||||
|
### @bot-whatsapp/bot
|
||||||
|
- [X] agregar export package
|
||||||
|
- [X] Posibilidad de en el capture meter todo un nuevo CTX de FLOW .addAnswer('Marca la opcion',{capture:true, join:CTX})
|
||||||
|
- [X] .addKeyword('1') no funciona con 1 caracter
|
||||||
|
- [X] sensitivy viene activado por defecto
|
||||||
|
- [X] fallback respuesta en hijo: Se puede colocar en option el ref de la answer fallback
|
||||||
|
- [X] Cuando Envian Sticket devuelve mensaje raro
|
||||||
|
- [x] addAnswer agregar delay
|
||||||
|
- [ ] colocar mensaje esperando conectando whatsapp (provider)
|
||||||
|
- [ ] createDatabase validar implementacion de funciones
|
||||||
|
- [ ] limitar caracteres de mensajes 4000
|
||||||
|
- [X] cuando envias numeros (5 o 1) se dispara el flujo
|
||||||
|
|
||||||
|
### @bot-whatsapp/database
|
||||||
|
- [X] agregar export package
|
||||||
|
- [X] __(doc):__ Video para explicar como implementar nuevos database
|
||||||
|
- [X] Mongo adapter
|
||||||
|
- [X] MySQL adapter
|
||||||
|
- [ ] JsonFile adapter
|
||||||
|
|
||||||
|
### @bot-whatsapp/provider
|
||||||
|
- [X] agregar export package
|
||||||
|
- [ ] __(doc):__ Video para explicar como implementar nuevos providers
|
||||||
|
- [X] WhatsappWeb provider enviar imagenes
|
||||||
|
- [X] WhatsappWeb provider enviar audio
|
||||||
|
- [X] WhatsappWeb botones (Tiene truco) github:leifermendez/whatsapp-web.js
|
||||||
|
- [ ] Twilio adapter
|
||||||
|
- [ ] Meta adapter
|
||||||
|
|
||||||
|
### @bot-whatsapp/cli
|
||||||
|
- [X] Hacer comando para crear `example-app`
|
||||||
|
|
||||||
|
|
||||||
|
### @bot-whatsapp/create-bot
|
||||||
|
- [ ]
|
||||||
|
|
||||||
|
### Starters
|
||||||
|
- [X] Base
|
||||||
|
- [X] Basico
|
||||||
|
- [ ] Enviando Imagen
|
||||||
|
- [ ] Enviando Botones
|
||||||
|
- [ ] Mezclando flujos hijos
|
||||||
|
|
||||||
|
### Extra
|
||||||
|
- [X] Crear CI mantener fork update https://stackoverflow.com/questions/23793062/can-forks-be-synced-automatically-in-github
|
||||||
50
__mocks__/env.js
Normal file
50
__mocks__/env.js
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
const MOCK_DB = require('../packages/database/src/mock')
|
||||||
|
const PROVIDER_DB = require('../packages/provider/src/mock')
|
||||||
|
|
||||||
|
class MOCK_FLOW {
|
||||||
|
allCallbacks = { ref: () => 1 }
|
||||||
|
flowSerialize = []
|
||||||
|
flowRaw = []
|
||||||
|
find = (arg) => {
|
||||||
|
if (arg) {
|
||||||
|
return [{ answer: 'answer', ref: 'ref' }]
|
||||||
|
} else {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
findBySerialize = () => ({})
|
||||||
|
findIndexByRef = () => 0
|
||||||
|
}
|
||||||
|
|
||||||
|
const cleaName = (name) => {
|
||||||
|
name = name.toLowerCase()
|
||||||
|
name = name.replaceAll(' ', '-')
|
||||||
|
name = name.replaceAll(':', '-')
|
||||||
|
name = name.replaceAll('"', '-')
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Preparar env para el test
|
||||||
|
* @param {*} context
|
||||||
|
*/
|
||||||
|
const setup = async (context) => {
|
||||||
|
const name = cleaName(`${context.__suite__}-${context.__test__}`)
|
||||||
|
const filename = `test-${name}.json`
|
||||||
|
context.provider = new PROVIDER_DB()
|
||||||
|
context.database = new MOCK_DB({ filename })
|
||||||
|
context.flow = new MOCK_FLOW()
|
||||||
|
await delay(10)
|
||||||
|
}
|
||||||
|
|
||||||
|
const clear = async (context) => {
|
||||||
|
context.provider = null
|
||||||
|
context.database = null
|
||||||
|
context.flow = null
|
||||||
|
}
|
||||||
|
|
||||||
|
function delay(ms) {
|
||||||
|
return new Promise((res) => setTimeout(res, ms))
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { setup, clear, delay }
|
||||||
21
__mocks__/mock.provider.js
Normal file
21
__mocks__/mock.provider.js
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
const ProviderClass = require('../packages/bot/provider/provider.class')
|
||||||
|
class MockProvider extends ProviderClass {
|
||||||
|
constructor() {
|
||||||
|
super()
|
||||||
|
}
|
||||||
|
|
||||||
|
delaySendMessage = (miliseconds, eventName, payload) =>
|
||||||
|
new Promise((res) =>
|
||||||
|
setTimeout(() => {
|
||||||
|
this.emit(eventName, payload)
|
||||||
|
res
|
||||||
|
}, miliseconds)
|
||||||
|
)
|
||||||
|
|
||||||
|
sendMessage = async (userId, message) => {
|
||||||
|
console.log(`Enviando... ${userId}, ${message}`)
|
||||||
|
return Promise.resolve({ userId, message })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = MockProvider
|
||||||
51
__test__/0.0.0-case.test.js
Normal file
51
__test__/0.0.0-case.test.js
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: Simple')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Responder a "hola"`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword('hola').addAnswer('Buenas!').addAnswer('Como vamos!')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(10)
|
||||||
|
assert.is('Buenas!', database.listHistory[0].answer)
|
||||||
|
assert.is('Como vamos!', database.listHistory[1].answer)
|
||||||
|
assert.is(undefined, database.listHistory[2])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase(`NO reponder a "pepe"`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword('hola').addAnswer('Buenas!').addAnswer('Como vamos!')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'pepe',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
|
||||||
|
assert.is(undefined, database.listHistory[0])
|
||||||
|
assert.is(undefined, database.listHistory[1])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
31
__test__/0.0.1-case.test.js
Normal file
31
__test__/0.0.1-case.test.js
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: Provider envia un location')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Responder a "CURRENT_LOCATION"`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword('#CURRENT_LOCATION#').addAnswer('Gracias por tu location')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: '#CURRENT_LOCATION#',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(200)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is('Gracias por tu location', getHistory[0])
|
||||||
|
assert.is(undefined, getHistory[1])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
51
__test__/0.1.0-case.test.js
Normal file
51
__test__/0.1.0-case.test.js
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: sensitive')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Responder a "ole" en minuscula`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword(['ola', 'ole'], { sensitive: true }).addAnswer('Bienvenido a la OLA')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'ole',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
|
||||||
|
assert.is('Bienvenido a la OLA', database.listHistory[0].answer)
|
||||||
|
assert.is(undefined, database.listHistory[1])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase(`NO Responder a "ole" en minuscula`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword(['ola', 'ole'], { sensitive: true }).addAnswer('Bienvenido a la OLA')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'OLE',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
|
||||||
|
assert.is(undefined, database.listHistory[0])
|
||||||
|
assert.is(undefined, database.listHistory[1])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
89
__test__/0.1.1-case.test.js
Normal file
89
__test__/0.1.1-case.test.js
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const fakeHTTP = async () => {
|
||||||
|
await delay(10)
|
||||||
|
}
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: hijos con callbacks')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Debe continuar el flujo del hijo`, async ({ database, provider }) => {
|
||||||
|
const flowCash = addKeyword('cash').addAnswer('Traeme los billetes! 😎')
|
||||||
|
|
||||||
|
const flowOnline = addKeyword('paypal')
|
||||||
|
.addAnswer('Voy generar un link de paypal *escribe algo*', { capture: true }, async (_, { flowDynamic }) => {
|
||||||
|
await fakeHTTP()
|
||||||
|
await flowDynamic('Esperate.... estoy generando esto toma su tiempo')
|
||||||
|
})
|
||||||
|
.addAnswer('Aqui lo tienes 😎😎', null, async (_, { flowDynamic }) => {
|
||||||
|
await fakeHTTP()
|
||||||
|
await flowDynamic('http://paypal.com')
|
||||||
|
})
|
||||||
|
.addAnswer('Apurate!')
|
||||||
|
|
||||||
|
const flujoPrincipal = addKeyword('hola')
|
||||||
|
.addAnswer('¿Como estas todo bien?')
|
||||||
|
.addAnswer('Espero que si')
|
||||||
|
.addAnswer('¿Cual es tu email?', { capture: true }, async (ctx, { fallBack }) => {
|
||||||
|
if (!ctx.body.includes('@')) {
|
||||||
|
return fallBack('Veo que no es um mail *bien*')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.addAnswer('Voy a validar tu email...', null, async (_, { flowDynamic }) => {
|
||||||
|
await fakeHTTP()
|
||||||
|
return flowDynamic('Email validado correctamten!')
|
||||||
|
})
|
||||||
|
.addAnswer('¿Como vas a pagar *paypal* o *cash*?', { capture: true }, async () => {}, [flowCash, flowOnline])
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
flow: createFlow([flujoPrincipal]),
|
||||||
|
provider,
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(30, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'test@test.com',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(60, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'paypal',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(90, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'continue!',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(800)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
|
||||||
|
assert.is('¿Como estas todo bien?', getHistory[0])
|
||||||
|
assert.is('Espero que si', getHistory[1])
|
||||||
|
assert.is('¿Cual es tu email?', getHistory[2])
|
||||||
|
assert.is('test@test.com', getHistory[3])
|
||||||
|
assert.is('Voy a validar tu email...', getHistory[4])
|
||||||
|
assert.is('Email validado correctamten!', getHistory[5])
|
||||||
|
assert.is('¿Como vas a pagar *paypal* o *cash*?', getHistory[6])
|
||||||
|
assert.is('paypal', getHistory[7])
|
||||||
|
assert.is('Voy generar un link de paypal *escribe algo*', getHistory[8])
|
||||||
|
assert.is('continue!', getHistory[9])
|
||||||
|
assert.is('Esperate.... estoy generando esto toma su tiempo', getHistory[10])
|
||||||
|
assert.is('Aqui lo tienes 😎😎', getHistory[11])
|
||||||
|
assert.is('http://paypal.com', getHistory[12])
|
||||||
|
assert.is('Apurate!', getHistory[13])
|
||||||
|
assert.is(undefined, getHistory[14])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
59
__test__/0.1.2-case.test.js
Normal file
59
__test__/0.1.2-case.test.js
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: regex')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Responder a una expresion regular`, async ({ database, provider }) => {
|
||||||
|
const REGEX_CREDIT_NUMBER = `/(^4[0-9]{12}(?:[0-9]{3})?$)|(^(?:5[1-5][0-9]{2}|222[1-9]|22[3-9][0-9]|2[3-6][0-9]{2}|27[01][0-9]|2720)[0-9]{12}$)|(3[47][0-9]{13})|(^3(?:0[0-5]|[68][0-9])[0-9]{11}$)|(^6(?:011|5[0-9]{2})[0-9]{12}$)|(^(?:2131|1800|35\d{3})\d{11}$)/gm`
|
||||||
|
|
||||||
|
const flow = addKeyword(REGEX_CREDIT_NUMBER, { regex: true })
|
||||||
|
.addAnswer(`Gracias por proporcionar un numero de tarjeta valido`)
|
||||||
|
.addAnswer('Fin!')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: '374245455400126',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
|
||||||
|
assert.is('Gracias por proporcionar un numero de tarjeta valido', database.listHistory[0].answer)
|
||||||
|
assert.is('Fin!', database.listHistory[1].answer)
|
||||||
|
assert.is(undefined, database.listHistory[2])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase(`NO Responder a una expresion regular`, async ({ database, provider }) => {
|
||||||
|
const REGEX_CREDIT_NUMBER = `/(^4[0-9]{12}(?:[0-9]{3})?$)|(^(?:5[1-5][0-9]{2}|222[1-9]|22[3-9][0-9]|2[3-6][0-9]{2}|27[01][0-9]|2720)[0-9]{12}$)|(3[47][0-9]{13})|(^3(?:0[0-5]|[68][0-9])[0-9]{11}$)|(^6(?:011|5[0-9]{2})[0-9]{12}$)|(^(?:2131|1800|35\d{3})\d{11}$)/gm`
|
||||||
|
const flow = addKeyword(REGEX_CREDIT_NUMBER, { regex: true })
|
||||||
|
.addAnswer(`Gracias por proporcionar un numero de tarjeta valido`)
|
||||||
|
.addAnswer('Fin!')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
|
||||||
|
assert.is(undefined, database.listHistory[0])
|
||||||
|
assert.is(undefined, database.listHistory[1])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
40
__test__/0.1.3-case.test.js
Normal file
40
__test__/0.1.3-case.test.js
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: capture')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Responder a "pregunta"`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword(['hola'])
|
||||||
|
.addAnswer(['Hola como estas?', '¿Cual es tu edad?'], { capture: true })
|
||||||
|
.addAnswer('Gracias por tu respuesta')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(10, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: '90',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
|
||||||
|
assert.is(['Hola como estas?', '¿Cual es tu edad?'].join('\n'), database.listHistory[0].answer)
|
||||||
|
assert.is('90', database.listHistory[1].answer)
|
||||||
|
assert.is('Gracias por tu respuesta', database.listHistory[2].answer)
|
||||||
|
assert.is(undefined, database.listHistory[3])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
162
__test__/0.1.4-case.test.js
Normal file
162
__test__/0.1.4-case.test.js
Normal file
@@ -0,0 +1,162 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const fakeHTTP = async (fakeData = []) => {
|
||||||
|
await delay(50)
|
||||||
|
const data = fakeData.map((u) => ({ body: `${u}` }))
|
||||||
|
return Promise.resolve(data)
|
||||||
|
}
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: flowDynamic')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Responder con mensajes asyncronos`, async ({ database, provider }) => {
|
||||||
|
const MOCK_VALUES = [
|
||||||
|
'Bienvenido te envio muchas marcas (5510)',
|
||||||
|
'Seleccione marca del auto a cotizar, con el *número* correspondiente',
|
||||||
|
'Seleccione la sub marca del auto a cotizar, con el *número* correspondiente:',
|
||||||
|
'Los precios rondan:',
|
||||||
|
]
|
||||||
|
const flow = addKeyword(['hola'])
|
||||||
|
.addAnswer(MOCK_VALUES[0], null, async (_, { flowDynamic }) => {
|
||||||
|
const data = await fakeHTTP(['Ford', 'GM', 'BMW'])
|
||||||
|
return flowDynamic(data)
|
||||||
|
})
|
||||||
|
.addAnswer(MOCK_VALUES[1], null, async (_, { flowDynamic }) => {
|
||||||
|
const data = await fakeHTTP(['Ranger', 'Explorer'])
|
||||||
|
return flowDynamic(data)
|
||||||
|
})
|
||||||
|
.addAnswer(MOCK_VALUES[2], null, async (_, { flowDynamic }) => {
|
||||||
|
const data = await fakeHTTP(['Usado', 'Nuevos'])
|
||||||
|
return flowDynamic(data)
|
||||||
|
})
|
||||||
|
.addAnswer(MOCK_VALUES[3], null, async (_, { flowDynamic }) => {
|
||||||
|
const data = await fakeHTTP(['1000', '2000', '3000'])
|
||||||
|
return flowDynamic(data)
|
||||||
|
})
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(1500)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is(MOCK_VALUES[0], getHistory[0])
|
||||||
|
|
||||||
|
//FlowDynamic
|
||||||
|
assert.is('Ford', getHistory[1])
|
||||||
|
assert.is('GM', getHistory[2])
|
||||||
|
assert.is('BMW', getHistory[3])
|
||||||
|
|
||||||
|
assert.is(MOCK_VALUES[1], getHistory[4])
|
||||||
|
|
||||||
|
//FlowDynamic
|
||||||
|
assert.is('Ranger', getHistory[5])
|
||||||
|
assert.is('Explorer', getHistory[6])
|
||||||
|
|
||||||
|
assert.is(MOCK_VALUES[2], getHistory[7])
|
||||||
|
|
||||||
|
//FlowDynamic
|
||||||
|
assert.is('Usado', getHistory[8])
|
||||||
|
assert.is('Nuevos', getHistory[9])
|
||||||
|
|
||||||
|
assert.is(MOCK_VALUES[3], getHistory[10])
|
||||||
|
|
||||||
|
//FlowDynamic
|
||||||
|
assert.is('1000', getHistory[11])
|
||||||
|
assert.is('2000', getHistory[12])
|
||||||
|
assert.is('3000', getHistory[13])
|
||||||
|
assert.is(undefined, getHistory[14])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase(`Responder con un "string"`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword(['hola'])
|
||||||
|
.addAnswer('Como vas?', null, async (_, { flowDynamic }) => {
|
||||||
|
return flowDynamic('Todo bien!')
|
||||||
|
})
|
||||||
|
.addAnswer('y vos?')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is('Como vas?', getHistory[0])
|
||||||
|
assert.is('Todo bien!', getHistory[1])
|
||||||
|
assert.is('y vos?', getHistory[2])
|
||||||
|
assert.is(undefined, getHistory[3])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase(`Responder con un "array"`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword(['hola'])
|
||||||
|
.addAnswer('Como vas?', null, async (_, { flowDynamic }) => {
|
||||||
|
return flowDynamic(['Todo bien!', 'trabajando'])
|
||||||
|
})
|
||||||
|
.addAnswer('y vos?')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is('Como vas?', getHistory[0])
|
||||||
|
assert.is('Todo bien!', getHistory[1])
|
||||||
|
assert.is('trabajando', getHistory[2])
|
||||||
|
assert.is('y vos?', getHistory[3])
|
||||||
|
assert.is(undefined, getHistory[4])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase(`Responder con un "object"`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword(['hola'])
|
||||||
|
.addAnswer('Como vas?', null, async (_, { flowDynamic }) => {
|
||||||
|
return flowDynamic([{ body: 'Todo bien!' }])
|
||||||
|
})
|
||||||
|
.addAnswer('y vos?')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is('Como vas?', getHistory[0])
|
||||||
|
assert.is('Todo bien!', getHistory[1])
|
||||||
|
assert.is('y vos?', getHistory[2])
|
||||||
|
assert.is(undefined, getHistory[3])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
167
__test__/0.1.5-case.test.js
Normal file
167
__test__/0.1.5-case.test.js
Normal file
@@ -0,0 +1,167 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const fakeHTTP = async (fakeData = []) => {
|
||||||
|
await delay(50)
|
||||||
|
const data = fakeData.map((u) => ({ body: `${u}` }))
|
||||||
|
return Promise.resolve(data)
|
||||||
|
}
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: endFlow')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Detener el flujo`, async ({ database, provider }) => {
|
||||||
|
const MOCK_VALUES = [
|
||||||
|
'Bienvenido te envio muchas marcas',
|
||||||
|
'Seleccione marca del auto a cotizar, con el *número* correspondiente',
|
||||||
|
'Seleccione la sub marca del auto a cotizar, con el *número* correspondiente:',
|
||||||
|
'Los precios rondan:',
|
||||||
|
]
|
||||||
|
const flow = addKeyword(['hola'])
|
||||||
|
.addAnswer(MOCK_VALUES[0], null, async (_, { flowDynamic }) => {
|
||||||
|
const data = await fakeHTTP(['Ford', 'GM', 'BMW'])
|
||||||
|
return flowDynamic(data)
|
||||||
|
})
|
||||||
|
.addAnswer(MOCK_VALUES[1], null, async (_, { endFlow }) => {
|
||||||
|
return endFlow()
|
||||||
|
})
|
||||||
|
.addAnswer(MOCK_VALUES[2])
|
||||||
|
.addAnswer(MOCK_VALUES[3], null, async (_, { flowDynamic }) => {
|
||||||
|
const data = await fakeHTTP(['1000', '2000', '3000'])
|
||||||
|
return flowDynamic(data)
|
||||||
|
})
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(900)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is(MOCK_VALUES[0], getHistory[0])
|
||||||
|
|
||||||
|
//FlowDynamic
|
||||||
|
assert.is('Ford', getHistory[1])
|
||||||
|
assert.is('GM', getHistory[2])
|
||||||
|
assert.is('BMW', getHistory[3])
|
||||||
|
|
||||||
|
assert.is(MOCK_VALUES[1], getHistory[4])
|
||||||
|
|
||||||
|
//FlowDynamic
|
||||||
|
assert.is(undefined, getHistory[5])
|
||||||
|
assert.is(undefined, getHistory[6])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase(`Detener el flujo flowDynamic`, async ({ database, provider }) => {
|
||||||
|
const flow = addKeyword(['hola'])
|
||||||
|
.addAnswer('Buenas!', null, async (_, { endFlow, flowDynamic }) => {
|
||||||
|
await flowDynamic('Continuamos...')
|
||||||
|
return endFlow()
|
||||||
|
})
|
||||||
|
.addAnswer('Como estas!')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(100)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is('Buenas!', getHistory[0])
|
||||||
|
assert.is('Continuamos...', getHistory[1])
|
||||||
|
assert.is(undefined, getHistory[2])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase(`flowDynamic con capture`, async ({ database, provider }) => {
|
||||||
|
const MOCK_VALUES = ['¿CUal es tu email?', 'Continuamos....', '¿Cual es tu edad?']
|
||||||
|
|
||||||
|
const flow = addKeyword(['hola'])
|
||||||
|
.addAnswer(
|
||||||
|
MOCK_VALUES[0],
|
||||||
|
{
|
||||||
|
capture: true,
|
||||||
|
},
|
||||||
|
async (ctx, { flowDynamic, fallBack }) => {
|
||||||
|
const validation = ctx.body.includes('@')
|
||||||
|
|
||||||
|
if (validation) {
|
||||||
|
const getDataFromApi = await fakeHTTP(['Gracias por tu email se ha validado de manera correcta'])
|
||||||
|
return flowDynamic(getDataFromApi)
|
||||||
|
}
|
||||||
|
return fallBack()
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.addAnswer(MOCK_VALUES[1])
|
||||||
|
.addAnswer(MOCK_VALUES[2], { capture: true }, async (ctx, { flowDynamic, fallBack }) => {
|
||||||
|
if (ctx.body !== '18') {
|
||||||
|
await delay(20)
|
||||||
|
return fallBack('Ups creo que no eres mayor de edad')
|
||||||
|
}
|
||||||
|
return flowDynamic('Bien tu edad es correcta!')
|
||||||
|
})
|
||||||
|
.addAnswer('Puedes pasar')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
provider,
|
||||||
|
flow: createFlow([flow]),
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(10, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'this is not email value',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(20, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'test@test.com',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(90, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: '20',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(200, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: '18',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(900)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is(MOCK_VALUES[0], getHistory[0])
|
||||||
|
assert.is('this is not email value', getHistory[1])
|
||||||
|
assert.is(MOCK_VALUES[0], getHistory[2])
|
||||||
|
assert.is('test@test.com', getHistory[3])
|
||||||
|
assert.is('Gracias por tu email se ha validado de manera correcta', getHistory[4])
|
||||||
|
assert.is(MOCK_VALUES[1], getHistory[5])
|
||||||
|
assert.is(MOCK_VALUES[2], getHistory[6])
|
||||||
|
assert.is('20', getHistory[7])
|
||||||
|
assert.is('Ups creo que no eres mayor de edad', getHistory[8])
|
||||||
|
assert.is('18', getHistory[9])
|
||||||
|
assert.is('Bien tu edad es correcta!', getHistory[10])
|
||||||
|
assert.is('Puedes pasar', getHistory[11])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
97
__test__/0.1.6-case.test.js
Normal file
97
__test__/0.1.6-case.test.js
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
const { suite } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { addKeyword, createBot, createFlow } = require('../packages/bot/index')
|
||||||
|
const { setup, clear, delay } = require('../__mocks__/env')
|
||||||
|
|
||||||
|
const suiteCase = suite('Flujo: manejo de estado')
|
||||||
|
|
||||||
|
suiteCase.before.each(setup)
|
||||||
|
suiteCase.after.each(clear)
|
||||||
|
|
||||||
|
suiteCase(`Debe retornar un mensaje resumen`, async ({ database, provider }) => {
|
||||||
|
let STATE_APP = {}
|
||||||
|
const MOCK_VALUES = ['¿Cual es tu nombre?', '¿Cual es tu edad?', 'Tu datos son:']
|
||||||
|
|
||||||
|
const flujoPrincipal = addKeyword(['hola'])
|
||||||
|
.addAnswer(
|
||||||
|
MOCK_VALUES[0],
|
||||||
|
{
|
||||||
|
capture: true,
|
||||||
|
},
|
||||||
|
async (ctx, { flowDynamic }) => {
|
||||||
|
STATE_APP[ctx.from] = { ...STATE_APP[ctx.from], name: ctx.body }
|
||||||
|
|
||||||
|
flowDynamic('Gracias por tu nombre!')
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.addAnswer(
|
||||||
|
MOCK_VALUES[1],
|
||||||
|
{
|
||||||
|
capture: true,
|
||||||
|
},
|
||||||
|
async (ctx, { flowDynamic }) => {
|
||||||
|
STATE_APP[ctx.from] = { ...STATE_APP[ctx.from], age: ctx.body }
|
||||||
|
|
||||||
|
await flowDynamic(`Gracias por tu edad! ${STATE_APP[ctx.from].name}`)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.addAnswer(MOCK_VALUES[2], null, async (ctx, { flowDynamic }) => {
|
||||||
|
flowDynamic(`Nombre: ${STATE_APP[ctx.from].name} Edad: ${STATE_APP[ctx.from].age}`)
|
||||||
|
})
|
||||||
|
.addAnswer('🤖🤖 Gracias por tu participacion')
|
||||||
|
|
||||||
|
createBot({
|
||||||
|
database,
|
||||||
|
flow: createFlow([flujoPrincipal]),
|
||||||
|
provider,
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(0, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(5, 'message', {
|
||||||
|
from: '001',
|
||||||
|
body: 'hola',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(10, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: 'Leifer',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(15, 'message', {
|
||||||
|
from: '000',
|
||||||
|
body: '90',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(20, 'message', {
|
||||||
|
from: '001',
|
||||||
|
body: 'Maria',
|
||||||
|
})
|
||||||
|
|
||||||
|
await provider.delaySendMessage(25, 'message', {
|
||||||
|
from: '001',
|
||||||
|
body: '100',
|
||||||
|
})
|
||||||
|
|
||||||
|
await delay(1000)
|
||||||
|
const getHistory = database.listHistory.map((i) => i.answer)
|
||||||
|
assert.is(MOCK_VALUES[0], getHistory[0])
|
||||||
|
assert.is('¿Cual es tu nombre?', getHistory[1])
|
||||||
|
assert.is('Leifer', getHistory[2])
|
||||||
|
assert.is('Gracias por tu nombre!', getHistory[3])
|
||||||
|
assert.is('¿Cual es tu edad?', getHistory[4])
|
||||||
|
assert.is('90', getHistory[5])
|
||||||
|
assert.is('Gracias por tu edad! Leifer', getHistory[6])
|
||||||
|
assert.is('Tu datos son:', getHistory[7])
|
||||||
|
assert.is('Nombre: Leifer Edad: 90', getHistory[8])
|
||||||
|
assert.is('🤖🤖 Gracias por tu participacion', getHistory[9])
|
||||||
|
assert.is('Maria', getHistory[10])
|
||||||
|
assert.is('Gracias por tu nombre!', getHistory[11])
|
||||||
|
assert.is('100', getHistory[12])
|
||||||
|
assert.is(undefined, getHistory[13])
|
||||||
|
})
|
||||||
|
|
||||||
|
suiteCase.run()
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
const dialogflow = require('@google-cloud/dialogflow');
|
|
||||||
const fs = require('fs')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Debes de tener tu archivo con el nombre "chatbot-account.json" en la raíz del proyecto
|
|
||||||
*/
|
|
||||||
|
|
||||||
const KEEP_DIALOG_FLOW = (process.env.KEEP_DIALOG_FLOW === 'true')
|
|
||||||
let PROJECID;
|
|
||||||
let CONFIGURATION;
|
|
||||||
let sessionClient;
|
|
||||||
|
|
||||||
const checkFileCredentials = () => {
|
|
||||||
if(!fs.existsSync(`${__dirname}/../chatbot-account.json`)){
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
const parseCredentials = JSON.parse(fs.readFileSync(`${__dirname}/../chatbot-account.json`));
|
|
||||||
PROJECID = parseCredentials.project_id;
|
|
||||||
CONFIGURATION = {
|
|
||||||
credentials: {
|
|
||||||
private_key: parseCredentials['private_key'],
|
|
||||||
client_email: parseCredentials['client_email']
|
|
||||||
}
|
|
||||||
}
|
|
||||||
sessionClient = new dialogflow.SessionsClient(CONFIGURATION);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create a new session
|
|
||||||
|
|
||||||
|
|
||||||
// Detect intent method
|
|
||||||
const detectIntent = async (queryText, waPhoneNumber) => {
|
|
||||||
let media = null;
|
|
||||||
const sessionId = KEEP_DIALOG_FLOW ? 1 : waPhoneNumber;
|
|
||||||
const sessionPath = sessionClient.projectAgentSessionPath(PROJECID, sessionId);
|
|
||||||
const languageCode = process.env.LANGUAGE
|
|
||||||
const request = {
|
|
||||||
session: sessionPath,
|
|
||||||
queryInput: {
|
|
||||||
text: {
|
|
||||||
text: queryText,
|
|
||||||
languageCode: languageCode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const responses = await sessionClient.detectIntent(request);
|
|
||||||
const [singleResponse] = responses;
|
|
||||||
const { queryResult } = singleResponse
|
|
||||||
const { intent } = queryResult || { intent: {} }
|
|
||||||
const parseIntent = intent['displayName'] || null
|
|
||||||
const parsePayload = queryResult['fulfillmentMessages'].find((a) => a.message === 'payload');
|
|
||||||
// console.log(singleResponse)
|
|
||||||
if (parsePayload && parsePayload.payload) {
|
|
||||||
const { fields } = parsePayload.payload
|
|
||||||
media = fields.media.stringValue || null
|
|
||||||
}
|
|
||||||
const customPayload = parsePayload ? parsePayload['payload'] : null
|
|
||||||
|
|
||||||
const parseData = {
|
|
||||||
replyMessage: queryResult.fulfillmentText,
|
|
||||||
media,
|
|
||||||
trigger: null
|
|
||||||
}
|
|
||||||
return parseData
|
|
||||||
}
|
|
||||||
|
|
||||||
const getDataIa = (message = '', sessionId = '', cb = () => { }) => {
|
|
||||||
detectIntent(message, sessionId).then((res) => {
|
|
||||||
cb(res)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
checkFileCredentials();
|
|
||||||
|
|
||||||
module.exports = { getDataIa }
|
|
||||||
@@ -1,90 +0,0 @@
|
|||||||
const { getData, getReply, saveMessageMysql } = require('./mysql')
|
|
||||||
const { saveMessageJson } = require('./jsonDb')
|
|
||||||
const { getDataIa } = require('./diaglogflow')
|
|
||||||
const stepsInitial = require('../flow/initial.json')
|
|
||||||
const stepsReponse = require('../flow/response.json')
|
|
||||||
|
|
||||||
const get = (message) => new Promise((resolve, reject) => {
|
|
||||||
/**
|
|
||||||
* Si no estas usando un gesto de base de datos
|
|
||||||
*/
|
|
||||||
|
|
||||||
if (process.env.DATABASE === 'none') {
|
|
||||||
const { key } = stepsInitial.find(k => k.keywords.includes(message)) || { key: null }
|
|
||||||
const response = key || null
|
|
||||||
resolve(response)
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* Si usas MYSQL
|
|
||||||
*/
|
|
||||||
if (process.env.DATABASE === 'mysql') {
|
|
||||||
getData(message, (dt) => {
|
|
||||||
resolve(dt)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
})
|
|
||||||
|
|
||||||
|
|
||||||
const reply = (step) => new Promise((resolve, reject) => {
|
|
||||||
/**
|
|
||||||
* Si no estas usando un gesto de base de datos
|
|
||||||
*/
|
|
||||||
if (process.env.DATABASE === 'none') {
|
|
||||||
let resData = { replyMessage: '', media: null, trigger: null }
|
|
||||||
const responseFind = stepsReponse[step] || {};
|
|
||||||
resData = {
|
|
||||||
...resData,
|
|
||||||
...responseFind,
|
|
||||||
replyMessage:responseFind.replyMessage.join('')}
|
|
||||||
resolve(resData);
|
|
||||||
return
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* Si usas MYSQL
|
|
||||||
*/
|
|
||||||
if (process.env.DATABASE === 'mysql') {
|
|
||||||
let resData = { replyMessage: '', media: null, trigger: null }
|
|
||||||
getReply(step, (dt) => {
|
|
||||||
resData = { ...resData, ...dt }
|
|
||||||
resolve(resData)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
const getIA = (message, sessionId) => new Promise((resolve, reject) => {
|
|
||||||
/**
|
|
||||||
* Si usas dialogflow
|
|
||||||
*/
|
|
||||||
if (process.env.DATABASE === 'dialogflow') {
|
|
||||||
let resData = { replyMessage: '', media: null, trigger: null }
|
|
||||||
getDataIa(message, sessionId, (dt) => {
|
|
||||||
resData = { ...resData, ...dt }
|
|
||||||
resolve(resData)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @param {*} message
|
|
||||||
* @param {*} date
|
|
||||||
* @param {*} trigger
|
|
||||||
* @param {*} number
|
|
||||||
* @returns
|
|
||||||
*/
|
|
||||||
const saveMessage = ( message, trigger, number ) => new Promise( async (resolve, reject) => {
|
|
||||||
switch ( process.env.DATABASE ) {
|
|
||||||
case 'mysql':
|
|
||||||
resolve( await saveMessageMysql( message, trigger, number ) )
|
|
||||||
break;
|
|
||||||
case 'none':
|
|
||||||
resolve( await saveMessageJson( message, trigger, number ) )
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
resolve(true)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
module.exports = { get, reply, getIA, saveMessage }
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
const Path = require('path')
|
|
||||||
const StormDB = require("stormdb");
|
|
||||||
const date = new Date().toISOString();
|
|
||||||
const saveMessageJson = (message, trigger, number) => new Promise( async(resolve,reject) =>{
|
|
||||||
try {
|
|
||||||
const engine = new StormDB.localFileEngine( Path.join(__dirname, `/../chats/${number}.json`) );
|
|
||||||
const db = new StormDB(engine);
|
|
||||||
// set default db value if db is empty
|
|
||||||
db.default({ messages: [] });
|
|
||||||
// add new users entry
|
|
||||||
db.get("messages").push({ message, date, trigger });
|
|
||||||
db.save();
|
|
||||||
resolve('Saved')
|
|
||||||
} catch (error) {
|
|
||||||
console.log(error)
|
|
||||||
reject(error)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
module.exports = { saveMessageJson }
|
|
||||||
@@ -1,74 +0,0 @@
|
|||||||
const {connection} = require('../config/mysql')
|
|
||||||
const DATABASE_NAME = process.env.SQL_DATABASE || 'db_test'
|
|
||||||
|
|
||||||
getData = (message = '', callback) => connection.query(
|
|
||||||
`SELECT * FROM ${DATABASE_NAME}.initial WHERE keywords LIKE '%${message}%' LIMIT 1`,
|
|
||||||
(error, results
|
|
||||||
) => {
|
|
||||||
const [response] = results
|
|
||||||
const key = response?.option_key || null
|
|
||||||
callback(key)
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
getReply = (option_key = '', callback) => connection.query(
|
|
||||||
`SELECT * FROM ${DATABASE_NAME}.response WHERE option_key = '${option_key}' LIMIT 1`,
|
|
||||||
(error, results
|
|
||||||
) => {
|
|
||||||
const [response] = results;
|
|
||||||
console.log(response)
|
|
||||||
const value = {
|
|
||||||
replyMessage:response?.replyMessage || '',
|
|
||||||
trigger:response?.trigger || '',
|
|
||||||
media:response?.media || ''
|
|
||||||
|
|
||||||
}
|
|
||||||
callback(value)
|
|
||||||
});
|
|
||||||
|
|
||||||
getMessages = ( number ) => new Promise((resolve,reejct) => {
|
|
||||||
try {
|
|
||||||
connection.query(
|
|
||||||
`SELECT * FROM ${DATABASE_NAME}.response WHERE number = '${number}'`, (error, results) => {
|
|
||||||
if(error) {
|
|
||||||
console.log(error)
|
|
||||||
}
|
|
||||||
const [response] = results;
|
|
||||||
console.log(response)
|
|
||||||
const value = {
|
|
||||||
replyMessage:response?.replyMessage || '',
|
|
||||||
trigger:response?.trigger || '',
|
|
||||||
media:response?.media || ''
|
|
||||||
}
|
|
||||||
resolve(value)
|
|
||||||
})
|
|
||||||
} catch (error) {
|
|
||||||
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
saveMessageMysql = ( message, date, trigger, number ) => new Promise((resolve,reejct) => {
|
|
||||||
try {
|
|
||||||
connection.query(
|
|
||||||
`INSERT INTO ${DATABASE_NAME}.messages `+"( `message`, `date`, `trigger`, `number`)"+` VALUES ('${message}','${date}','${trigger}', '${number}')` , (error, results) => {
|
|
||||||
if(error) {
|
|
||||||
//TODO esta parte es mejor incluirla directamente en el archivo .sql template
|
|
||||||
console.log('DEBES DE CREAR LA TABLA DE MESSAGE')
|
|
||||||
// if( error.code === 'ER_NO_SUCH_TABLE' ){
|
|
||||||
// connection.query( `CREATE TABLE ${DATABASE_NAME}.messages `+"( `date` DATE NOT NULL , `message` VARCHAR(450) NOT NULL , `trigger` VARCHAR(450) NOT NULL , `number` VARCHAR(50) NOT NULL ) ENGINE = InnoDB", async (error, results) => {
|
|
||||||
// setTimeout( async () => {
|
|
||||||
// return resolve( await this.saveMessageMysql( message, date, trigger, number ) )
|
|
||||||
// }, 150)
|
|
||||||
// })
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
console.log('Saved')
|
|
||||||
console.log( results )
|
|
||||||
resolve(results)
|
|
||||||
})
|
|
||||||
} catch (error) {
|
|
||||||
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
module.exports = {getData, getReply, saveMessageMysql}
|
|
||||||
171
app.js
171
app.js
@@ -1,171 +0,0 @@
|
|||||||
/**
|
|
||||||
* ⚡⚡⚡ DECLARAMOS LAS LIBRERIAS y CONSTANTES A USAR! ⚡⚡⚡
|
|
||||||
*/
|
|
||||||
require('dotenv').config()
|
|
||||||
const fs = require('fs');
|
|
||||||
const express = require('express');
|
|
||||||
const cors = require('cors')
|
|
||||||
const qrcode = require('qrcode-terminal');
|
|
||||||
const { Client, LocalAuth } = require('whatsapp-web.js');
|
|
||||||
const mysqlConnection = require('./config/mysql')
|
|
||||||
const { middlewareClient } = require('./middleware/client')
|
|
||||||
const { generateImage, cleanNumber, checkEnvFile, createClient, isValidNumber } = require('./controllers/handle')
|
|
||||||
const { connectionReady, connectionLost } = require('./controllers/connection')
|
|
||||||
const { saveMedia } = require('./controllers/save')
|
|
||||||
const { getMessages, responseMessages, bothResponse } = require('./controllers/flows')
|
|
||||||
const { sendMedia, sendMessage, lastTrigger, sendMessageButton, readChat } = require('./controllers/send')
|
|
||||||
const app = express();
|
|
||||||
app.use(cors())
|
|
||||||
app.use(express.json())
|
|
||||||
const MULTI_DEVICE = process.env.MULTI_DEVICE || 'true';
|
|
||||||
const server = require('http').Server(app)
|
|
||||||
|
|
||||||
const port = process.env.PORT || 3000
|
|
||||||
var client;
|
|
||||||
app.use('/', require('./routes/web'))
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Escuchamos cuando entre un mensaje
|
|
||||||
*/
|
|
||||||
const listenMessage = () => client.on('message', async msg => {
|
|
||||||
const { from, body, hasMedia } = msg;
|
|
||||||
|
|
||||||
if (!isValidNumber(from)) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Este bug lo reporto Lucas Aldeco Brescia para evitar que se publiquen estados
|
|
||||||
if (from === 'status@broadcast') {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
message = body.toLowerCase();
|
|
||||||
console.log('BODY', message)
|
|
||||||
const number = cleanNumber(from)
|
|
||||||
await readChat(number, message)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Guardamos el archivo multimedia que envia
|
|
||||||
*/
|
|
||||||
if (process.env.SAVE_MEDIA && hasMedia) {
|
|
||||||
const media = await msg.downloadMedia();
|
|
||||||
saveMedia(media);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Si estas usando dialogflow solo manejamos una funcion todo es IA
|
|
||||||
*/
|
|
||||||
|
|
||||||
if (process.env.DATABASE === 'dialogflow') {
|
|
||||||
if (!message.length) return;
|
|
||||||
const response = await bothResponse(message, number);
|
|
||||||
await sendMessage(client, from, response.replyMessage);
|
|
||||||
if (response.media) {
|
|
||||||
sendMedia(client, from, response.media);
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Ver si viene de un paso anterior
|
|
||||||
* Aqui podemos ir agregando más pasos
|
|
||||||
* a tu gusto!
|
|
||||||
*/
|
|
||||||
|
|
||||||
const lastStep = await lastTrigger(from) || null;
|
|
||||||
if (lastStep) {
|
|
||||||
const response = await responseMessages(lastStep)
|
|
||||||
await sendMessage(client, from, response.replyMessage);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Respondemos al primero paso si encuentra palabras clave
|
|
||||||
*/
|
|
||||||
const step = await getMessages(message);
|
|
||||||
|
|
||||||
if (step) {
|
|
||||||
const response = await responseMessages(step);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Si quieres enviar botones
|
|
||||||
*/
|
|
||||||
|
|
||||||
await sendMessage(client, from, response.replyMessage, response.trigger);
|
|
||||||
|
|
||||||
if (response.hasOwnProperty('actions')) {
|
|
||||||
const { actions } = response;
|
|
||||||
await sendMessageButton(client, from, null, actions);
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!response.delay && response.media) {
|
|
||||||
sendMedia(client, from, response.media);
|
|
||||||
}
|
|
||||||
if (response.delay && response.media) {
|
|
||||||
setTimeout(() => {
|
|
||||||
sendMedia(client, from, response.media);
|
|
||||||
}, response.delay)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
//Si quieres tener un mensaje por defecto
|
|
||||||
if (process.env.DEFAULT_MESSAGE === 'true') {
|
|
||||||
const response = await responseMessages('DEFAULT')
|
|
||||||
await sendMessage(client, from, response.replyMessage, response.trigger);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Si quieres enviar botones
|
|
||||||
*/
|
|
||||||
if (response.hasOwnProperty('actions')) {
|
|
||||||
const { actions } = response;
|
|
||||||
await sendMessageButton(client, from, null, actions);
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
client = new Client({
|
|
||||||
authStrategy: new LocalAuth(),
|
|
||||||
puppeteer: { headless: true }
|
|
||||||
});
|
|
||||||
|
|
||||||
client.on('qr', qr => generateImage(qr, () => {
|
|
||||||
qrcode.generate(qr, { small: true });
|
|
||||||
|
|
||||||
console.log(`Ver QR http://localhost:${port}/qr`)
|
|
||||||
socketEvents.sendQR(qr)
|
|
||||||
}))
|
|
||||||
|
|
||||||
client.on('ready', (a) => {
|
|
||||||
connectionReady()
|
|
||||||
listenMessage()
|
|
||||||
// socketEvents.sendStatus(client)
|
|
||||||
});
|
|
||||||
|
|
||||||
client.on('auth_failure', (e) => {
|
|
||||||
// console.log(e)
|
|
||||||
// connectionLost()
|
|
||||||
});
|
|
||||||
|
|
||||||
client.on('authenticated', () => {
|
|
||||||
console.log('AUTHENTICATED');
|
|
||||||
});
|
|
||||||
|
|
||||||
client.initialize();
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Verificamos si tienes un gesto de db
|
|
||||||
*/
|
|
||||||
|
|
||||||
if (process.env.DATABASE === 'mysql') {
|
|
||||||
mysqlConnection.connect()
|
|
||||||
}
|
|
||||||
|
|
||||||
server.listen(port, () => {
|
|
||||||
console.log(`El server esta listo por el puerto ${port}`);
|
|
||||||
})
|
|
||||||
checkEnvFile();
|
|
||||||
35
app.json
35
app.json
@@ -1,35 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "Chatbot Whatsapp (Leifer Mendez)",
|
|
||||||
"description": "El siguiente proyecto se realizó con fines educativos para el canal de Youtube (Leifer Mendez) donde aprendemos como usando node.js podemos crear un chatbot increíble que además le agregamos inteligencia artificial gracias al servicio de dialogflow",
|
|
||||||
"repository": "https://github.com/leifermendez/bot-whatsapp",
|
|
||||||
"logo": "https://avatars0.githubusercontent.com/u/15802366?s=460&u=77ec7ef359e8ed842aef769693f1675c0ed460fd&v=4",
|
|
||||||
"keywords": [
|
|
||||||
"nodejs",
|
|
||||||
"whatsapp",
|
|
||||||
"bot",
|
|
||||||
"chatbot",
|
|
||||||
"dialogflow"
|
|
||||||
],
|
|
||||||
"addons": [
|
|
||||||
],
|
|
||||||
"buildpacks": [
|
|
||||||
{
|
|
||||||
"url": "heroku/nodejs"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"url": "https://github.com/jontewks/puppeteer-heroku-buildpack"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"env": {
|
|
||||||
"SAVE_MEDIA": "false",
|
|
||||||
"DATABASE": {
|
|
||||||
"description": "'none', 'mysql', 'dialogflow' por defecto 'none' Puedes usar alguna de los siguientes opciones. Pero antes debes de saber como funciona y eso lo explico en el video. Puedes obtener más información https://github.com/leifermendez/bot-whatsapp/blob/main/README.md",
|
|
||||||
"value": "none"
|
|
||||||
},
|
|
||||||
"LANGUAGE": "es",
|
|
||||||
"SQL_HOST":"your_host",
|
|
||||||
"SQL_USER":"your_user",
|
|
||||||
"SQL_PASS":"your_password",
|
|
||||||
"SQL_DATABASE":"your_database"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
70
changelog.config.js
Normal file
70
changelog.config.js
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
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,13 +0,0 @@
|
|||||||
{
|
|
||||||
"type": "",
|
|
||||||
"project_id": "",
|
|
||||||
"private_key_id": "",
|
|
||||||
"private_key":"",
|
|
||||||
"client_email": "",
|
|
||||||
"client_id": "",
|
|
||||||
"auth_uri": "",
|
|
||||||
"token_uri": "",
|
|
||||||
"auth_provider_x509_cert_url": "",
|
|
||||||
"client_x509_cert_url":""
|
|
||||||
}
|
|
||||||
|
|
||||||
1
commitlint.config.js
Normal file
1
commitlint.config.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
module.exports = { extends: ['@commitlint/config-conventional'] }
|
||||||
8
config/banner.rollup.json
Normal file
8
config/banner.rollup.json
Normal 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",
|
||||||
|
"*/"
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
const mysql = require('mysql');
|
|
||||||
const connection = mysql.createConnection({
|
|
||||||
host : process.env.SQL_HOST || 'localhost',
|
|
||||||
user : process.env.SQL_USER || 'root',
|
|
||||||
password : process.env.SQL_PASS || '',
|
|
||||||
database : process.env.SQL_DATABASE || 'pruebas'
|
|
||||||
});
|
|
||||||
|
|
||||||
const connect = () => connection.connect(function(err) {
|
|
||||||
if (err) {
|
|
||||||
console.error('error connecting: ' + err.stack);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('Conexion correcta con tu base de datos MySQL')
|
|
||||||
});
|
|
||||||
|
|
||||||
module.exports = {connect, connection}
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
const connectionReady = (cb = () =>{}) => {
|
|
||||||
console.log('Listo para escuchas mensajes')
|
|
||||||
console.log('Client is ready!');
|
|
||||||
console.log('🔴 escribe: hola');
|
|
||||||
cb()
|
|
||||||
}
|
|
||||||
|
|
||||||
const connectionLost = (cb = () =>{}) => {
|
|
||||||
console.log('** Error de autentificacion vuelve a generar el QRCODE (Borrar el archivo session.json) **');
|
|
||||||
cb()
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
module.exports = {connectionReady, connectionLost}
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
const {get, reply, getIA} = require('../adapter')
|
|
||||||
const {saveExternalFile, checkIsUrl} = require('./handle')
|
|
||||||
|
|
||||||
const getMessages = async (message) => {
|
|
||||||
const data = await get(message)
|
|
||||||
return data
|
|
||||||
}
|
|
||||||
|
|
||||||
const responseMessages = async (step) => {
|
|
||||||
const data = await reply(step)
|
|
||||||
if(data && data.media){
|
|
||||||
const file = checkIsUrl(data.media) ? await saveExternalFile(data.media) : data.media;
|
|
||||||
return {...data,...{media:file}}
|
|
||||||
}
|
|
||||||
return data
|
|
||||||
}
|
|
||||||
|
|
||||||
const bothResponse = async (message, sessionId) => {
|
|
||||||
const data = await getIA(message, sessionId)
|
|
||||||
if(data && data.media){
|
|
||||||
const file = await saveExternalFile(data.media)
|
|
||||||
return {...data,...{media:file}}
|
|
||||||
}
|
|
||||||
return data
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
module.exports = { getMessages, responseMessages, bothResponse }
|
|
||||||
@@ -1,83 +0,0 @@
|
|||||||
const { Client, LegacySessionAuth, LocalAuth } = require('whatsapp-web.js');
|
|
||||||
const http = require('http'); // or 'https' for https:// URLs
|
|
||||||
const https = require('https'); // or 'https' for https:// URLs
|
|
||||||
const fs = require('fs');
|
|
||||||
const qr = require('qr-image')
|
|
||||||
|
|
||||||
const MULTI_DEVICE = process.env.MULTI_DEVICE || 'true';
|
|
||||||
|
|
||||||
const cleanNumber = (number) => {
|
|
||||||
number = number.replace('@c.us', '');
|
|
||||||
number = `${number}@c.us`;
|
|
||||||
return number
|
|
||||||
}
|
|
||||||
|
|
||||||
const saveExternalFile = (url) => new Promise((resolve, reject) => {
|
|
||||||
const ext = url.split('.').pop()
|
|
||||||
const checkProtocol = url.split('/').includes('https:');
|
|
||||||
const handleHttp = checkProtocol ? https : http;
|
|
||||||
const name = `${Date.now()}.${ext}`;
|
|
||||||
const file = fs.createWriteStream(`${__dirname}/../mediaSend/${name}`);
|
|
||||||
console.log(url)
|
|
||||||
handleHttp.get(url, function(response) {
|
|
||||||
response.pipe(file);
|
|
||||||
file.on('finish', function() {
|
|
||||||
file.close(); // close() is async, call cb after close completes.
|
|
||||||
resolve(name)
|
|
||||||
});
|
|
||||||
file.on('error', function() {
|
|
||||||
console.log('errro')
|
|
||||||
file.close(); // close() is async, call cb after close completes.
|
|
||||||
resolve(null)
|
|
||||||
});
|
|
||||||
});
|
|
||||||
})
|
|
||||||
|
|
||||||
const checkIsUrl = (path) => {
|
|
||||||
try{
|
|
||||||
regex = /^(http(s)?:\/\/)[\w.-]+(?:\.[\w\.-]+)+[\w\-\._~:/?#[\]@!\$&'\(\)\*\+,;=.]+$/i;
|
|
||||||
match = path.match(regex);
|
|
||||||
return match[0]
|
|
||||||
}catch(e){
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const generateImage = (base64, cb = () => {}) => {
|
|
||||||
let qr_svg = qr.image(base64, { type: 'svg', margin: 4 });
|
|
||||||
qr_svg.pipe(require('fs').createWriteStream('./mediaSend/qr-code.svg'));
|
|
||||||
console.log(`⚡ Recuerda que el QR se actualiza cada minuto ⚡'`);
|
|
||||||
console.log(`⚡ Actualiza F5 el navegador para mantener el mejor QR⚡`);
|
|
||||||
cb()
|
|
||||||
}
|
|
||||||
|
|
||||||
const checkEnvFile = () => {
|
|
||||||
const pathEnv = `${__dirname}/../.env`;
|
|
||||||
const isExist = fs.existsSync(pathEnv);
|
|
||||||
if(!isExist){
|
|
||||||
console.log(`🆗 ATENCION! 🆗 te falta crear tu archivo .env de lo contrario no funcionara`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @param {*} session
|
|
||||||
* @param {*} cb
|
|
||||||
*/
|
|
||||||
const createClient = () => {
|
|
||||||
client = new Client({
|
|
||||||
authStrategy: new LocalAuth(
|
|
||||||
{dataPath: './sessions/',
|
|
||||||
clientId: 'bot'}),
|
|
||||||
puppeteer: { headless: false }
|
|
||||||
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const isValidNumber = (rawNumber) => {
|
|
||||||
const regexGroup = /\@g.us\b/gm;
|
|
||||||
const exist = rawNumber.match(regexGroup);
|
|
||||||
return !exist
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = {cleanNumber, saveExternalFile, generateImage, checkIsUrl, checkEnvFile, createClient, isValidNumber}
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
const mimeDb = require('mime-db')
|
|
||||||
const fs = require('fs')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Guardamos archivos multimedia que nuestro cliente nos envie!
|
|
||||||
* @param {*} media
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
const saveMedia = (media) => {
|
|
||||||
const extensionProcess = mimeDb[media.mimetype]
|
|
||||||
const ext = extensionProcess.extensions[0]
|
|
||||||
fs.writeFile(`./media/${Date.now()}.${ext}`, media.data, { encoding: 'base64' }, function (err) {
|
|
||||||
console.log('** Archivo Media Guardado **');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = {saveMedia}
|
|
||||||
@@ -1,113 +0,0 @@
|
|||||||
|
|
||||||
const ExcelJS = require('exceljs');
|
|
||||||
const moment = require('moment');
|
|
||||||
const fs = require('fs');
|
|
||||||
const { MessageMedia, Buttons } = require('whatsapp-web.js');
|
|
||||||
const { cleanNumber } = require('./handle')
|
|
||||||
const DELAY_TIME = 170; //ms
|
|
||||||
const DIR_MEDIA = `${__dirname}/../mediaSend`;
|
|
||||||
// import { Low, JSONFile } from 'lowdb'
|
|
||||||
// import { join } from 'path'
|
|
||||||
const { saveMessage } = require('../adapter')
|
|
||||||
/**
|
|
||||||
* Enviamos archivos multimedia a nuestro cliente
|
|
||||||
* @param {*} number
|
|
||||||
* @param {*} fileName
|
|
||||||
*/
|
|
||||||
|
|
||||||
const sendMedia = (client, number = null, fileName = null) => {
|
|
||||||
if(!client) return cosnole.error("El objeto cliente no está definido.");
|
|
||||||
try {
|
|
||||||
number = cleanNumber(number || 0)
|
|
||||||
const file = `${DIR_MEDIA}/${fileName}`;
|
|
||||||
if (fs.existsSync(file)) {
|
|
||||||
const media = MessageMedia.fromFilePath(file);
|
|
||||||
client.sendMessage(number, media, { sendAudioAsVoice: true });
|
|
||||||
}
|
|
||||||
} catch(e) {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Enviamos archivos como notas de voz
|
|
||||||
* @param {*} number
|
|
||||||
* @param {*} fileName
|
|
||||||
*/
|
|
||||||
|
|
||||||
const sendMediaVoiceNote = (client, number = null, fileName = null) => {
|
|
||||||
if(!client) return cosnole.error("El objeto cliente no está definido.");
|
|
||||||
try {
|
|
||||||
number = cleanNumber(number || 0)
|
|
||||||
const file = `${DIR_MEDIA}/${fileName}`;
|
|
||||||
if (fs.existsSync(file)) {
|
|
||||||
const media = MessageMedia.fromFilePath(file);
|
|
||||||
client.sendMessage(number, media ,{ sendAudioAsVoice: true });
|
|
||||||
|
|
||||||
}
|
|
||||||
}catch(e) {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* Enviamos un mensaje simple (texto) a nuestro cliente
|
|
||||||
* @param {*} number
|
|
||||||
*/
|
|
||||||
const sendMessage = async (client, number = null, text = null, trigger = null) => {
|
|
||||||
setTimeout(async () => {
|
|
||||||
number = cleanNumber(number)
|
|
||||||
const message = text
|
|
||||||
client.sendMessage(number, message);
|
|
||||||
await readChat(number, message, trigger)
|
|
||||||
console.log(`⚡⚡⚡ Enviando mensajes....`);
|
|
||||||
},DELAY_TIME)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Enviamos un mensaje con buttons a nuestro cliente
|
|
||||||
* @param {*} number
|
|
||||||
*/
|
|
||||||
const sendMessageButton = async (client, number = null, text = null, actionButtons) => {
|
|
||||||
number = cleanNumber(number)
|
|
||||||
const { title = null, message = null, footer = null, buttons = [] } = actionButtons;
|
|
||||||
let button = new Buttons(message,[...buttons], title, footer);
|
|
||||||
client.sendMessage(number, button);
|
|
||||||
|
|
||||||
console.log(`⚡⚡⚡ Enviando mensajes....`);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Opte
|
|
||||||
*/
|
|
||||||
const lastTrigger = (number) => new Promise((resolve, reject) => {
|
|
||||||
number = cleanNumber(number)
|
|
||||||
const pathExcel = `${__dirname}/../chats/${number}.xlsx`;
|
|
||||||
const workbook = new ExcelJS.Workbook();
|
|
||||||
if (fs.existsSync(pathExcel)) {
|
|
||||||
workbook.xlsx.readFile(pathExcel)
|
|
||||||
.then(() => {
|
|
||||||
const worksheet = workbook.getWorksheet(1);
|
|
||||||
const lastRow = worksheet.lastRow;
|
|
||||||
const getRowPrevStep = worksheet.getRow(lastRow.number);
|
|
||||||
const lastStep = getRowPrevStep.getCell('C').value;
|
|
||||||
resolve(lastStep)
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
resolve(null)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Guardar historial de conversacion
|
|
||||||
* @param {*} number
|
|
||||||
* @param {*} message
|
|
||||||
*/
|
|
||||||
const readChat = async (number, message, trigger = null) => {
|
|
||||||
number = cleanNumber(number)
|
|
||||||
await saveMessage( message, trigger, number )
|
|
||||||
console.log('Saved')
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = { sendMessage, sendMedia, lastTrigger, sendMessageButton, readChat, sendMediaVoiceNote }
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
module.exports = (socket) => {
|
|
||||||
return {
|
|
||||||
sendQR:(qr) => {
|
|
||||||
socket.emit('connection_qr',{
|
|
||||||
qr
|
|
||||||
})
|
|
||||||
},
|
|
||||||
sendStatus:() => {
|
|
||||||
socket.emit('connection_status',{
|
|
||||||
a:1
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
const fs = require('fs')
|
|
||||||
const { sendMessage } = require('../controllers/send')
|
|
||||||
|
|
||||||
const sendMessagePost = (req, res) => {
|
|
||||||
console.log('asdasdasdasdasd')
|
|
||||||
const { message, number } = req.body
|
|
||||||
const client = req.clientWs || null;
|
|
||||||
sendMessage(client, number, message)
|
|
||||||
res.send({ status: 'Enviado!' })
|
|
||||||
}
|
|
||||||
|
|
||||||
const getQr = (req, res) => {
|
|
||||||
res.writeHead(200, { 'content-type': 'image/svg+xml' });
|
|
||||||
fs.createReadStream(`${__dirname}/../mediaSend/qr-code.svg`).pipe(res);
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = { sendMessagePost, getQr }
|
|
||||||
25
docker-compose.yml
Normal file
25
docker-compose.yml
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
version: '3.3'
|
||||||
|
|
||||||
|
services:
|
||||||
|
mongo:
|
||||||
|
image: mongo
|
||||||
|
container_name: app_enviroment
|
||||||
|
restart: always
|
||||||
|
ports:
|
||||||
|
- '27019:27017'
|
||||||
|
environment:
|
||||||
|
MONGO_INITDB_DATABASE: bot
|
||||||
|
expose:
|
||||||
|
- 27019
|
||||||
|
mysql:
|
||||||
|
image: mysql
|
||||||
|
command: --default-authentication-plugin=mysql_native_password
|
||||||
|
restart: always
|
||||||
|
environment:
|
||||||
|
MYSQL_ROOT_PASSWORD: example
|
||||||
|
MYSQL_DATABASE: bot
|
||||||
|
container_name: app_mysql
|
||||||
|
ports:
|
||||||
|
- '3306:3306'
|
||||||
|
expose:
|
||||||
|
- 3306
|
||||||
@@ -1,93 +0,0 @@
|
|||||||
[
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"hola",
|
|
||||||
"hola!",
|
|
||||||
"ola",
|
|
||||||
"ole",
|
|
||||||
"inicio",
|
|
||||||
"welcome",
|
|
||||||
"buenos días",
|
|
||||||
"buenas tardes",
|
|
||||||
"buenas noches",
|
|
||||||
"me dieron este número",
|
|
||||||
"venden a crédito",
|
|
||||||
"quisiera saber si venden",
|
|
||||||
"necesito saber"
|
|
||||||
],
|
|
||||||
"key": "STEP_1"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"cursos",
|
|
||||||
"info",
|
|
||||||
"curso" ],
|
|
||||||
"key": "STEP_2"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"angular"
|
|
||||||
],
|
|
||||||
"key": "STEP_2_1"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"node"
|
|
||||||
],
|
|
||||||
"key": "STEP_2_2"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"ngrx"
|
|
||||||
],
|
|
||||||
"key": "STEP_2_3"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"aws"
|
|
||||||
],
|
|
||||||
"key": "STEP_2_4"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"asesor",
|
|
||||||
"asesores",
|
|
||||||
"Vendedor",
|
|
||||||
"cobrador"
|
|
||||||
],
|
|
||||||
"key": "STEP_3"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"muchas gracias",
|
|
||||||
"ok",
|
|
||||||
"gracias",
|
|
||||||
"vale gracias"
|
|
||||||
],
|
|
||||||
"key": "STEP_4"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"youtube"
|
|
||||||
],
|
|
||||||
"key": "STEP_5"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"VER_CURSOS"
|
|
||||||
],
|
|
||||||
"key": "STEP_6"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"telegram"
|
|
||||||
],
|
|
||||||
"key": "STEP_7"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"keywords": [
|
|
||||||
"audio"
|
|
||||||
],
|
|
||||||
"key": "STEP_8"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
@@ -1,148 +0,0 @@
|
|||||||
{
|
|
||||||
"DEFAULT":{
|
|
||||||
"replyMessage":[
|
|
||||||
"*Esta respuesta es un respuesta default* cuando no se consigue una palabra clave \n",
|
|
||||||
"la puedes desactivar en tu archivo .env DEFAULT_MESSAGE=false \n",
|
|
||||||
"tambien te quiero recordar que si presentas algun error pasarte por el repositorio \n",
|
|
||||||
"https://github.com/leifermendez/bot-whatsapp#chatbot-whatsapp-opensource \n",
|
|
||||||
"y recuerda tener la ultima versión del proyecto \n\n",
|
|
||||||
"Prueba escribiendo *hola* \n"
|
|
||||||
],
|
|
||||||
"media":null,
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_0":{
|
|
||||||
"replyMessage":[
|
|
||||||
"El flujo ha finalizado \n",
|
|
||||||
"pero puedes ver todo el codigo de este \n",
|
|
||||||
"repositorio en https://github.com/leifermendez/bot-whatsapp.git"
|
|
||||||
],
|
|
||||||
"media":null,
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_1":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Hola! y✌️ Bienvenido a este 🤖 CHATBOT de Whatsapp, lo primero \n",
|
|
||||||
"decirte que mi nombre es *Leifer Mendez*😎 \n",
|
|
||||||
"\n Si necesitas ver más info sobre las capacitacion tecnicas ",
|
|
||||||
"escribe *cursos* o *info* o escribe *audio*"
|
|
||||||
],
|
|
||||||
"media":"https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif",
|
|
||||||
"trigger":null,
|
|
||||||
"actions":{
|
|
||||||
"title":"¿Que te interesa ver?",
|
|
||||||
"message":"Recuerda todo este contenido es gratis y estaria genial que me siguas!",
|
|
||||||
"footer":"Gracias",
|
|
||||||
"buttons":[
|
|
||||||
{"body":"Cursos"},
|
|
||||||
{"body":"Youtube"},
|
|
||||||
{"body":"Telegram"}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"STEP_2":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Perfecto, te voy a pasar la lista ",
|
|
||||||
"de los temas que tengo y un breve video 🙂🤖 \n\n",
|
|
||||||
"*Angular* Basico (Pago) \n",
|
|
||||||
"*Angular* Basico (Gratis) \n",
|
|
||||||
"*Node* Basico (Gratis) \n",
|
|
||||||
"*NGRX* Basico (Gratis) \n",
|
|
||||||
"*AWS* Basico (Pago) \n\n",
|
|
||||||
"Escribe la palabra del tema que te interese \n"
|
|
||||||
],
|
|
||||||
"media":"https://i.giphy.com/media/5J5gN0WUk0VToHaK2p/giphy-downsized.gif",
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_2_1":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Si te interesa Angular tienes disponible \n",
|
|
||||||
"*(Gratis)* https://bit.ly/367tJ32 \n\n",
|
|
||||||
"*(Pago)* https://link.codigoencasa.com/PROMO-INICIAL \n\n",
|
|
||||||
"*(Pago)* https://link.codigoencasa.com/ANGULAR-BASICO-EDTEAM \n\n",
|
|
||||||
"😎😎😎"
|
|
||||||
],
|
|
||||||
"media":"https://i.imgur.com/Q0a5UQI.jpg",
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_2_2":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Si te interesa NODE tienes disponible \n",
|
|
||||||
"*(Gratis)* https://bit.ly/3od1Bl6 \n\n",
|
|
||||||
"Espero pronto tener más material disponible",
|
|
||||||
"🤖"
|
|
||||||
],
|
|
||||||
"media":null,
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_2_3":{
|
|
||||||
"replyMessage":[
|
|
||||||
"NGRX para manejar estados en Angular \n",
|
|
||||||
"*(Gratis)* https://bit.ly/ngrx-desde-cero \n",
|
|
||||||
"A darle! 😮"
|
|
||||||
],
|
|
||||||
"media":null,
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_2_4":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Muy bien AWS esta pronto a salir pre-registrate aquí \n",
|
|
||||||
"*(Pre-registro)* https://link.codigoencasa.com/AWS-BASICO-INVITACION \n",
|
|
||||||
"😮😮"
|
|
||||||
],
|
|
||||||
"media":null,
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_3":{
|
|
||||||
"replyMessage":[
|
|
||||||
"¿Ok cual curso de intereso? \n",
|
|
||||||
"*angular* , *node*, *ngrx*, *aws*"
|
|
||||||
],
|
|
||||||
"media":null,
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_4":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Gracias a ti! \n"
|
|
||||||
],
|
|
||||||
"media":"https://media4.giphy.com/media/hur0SFIU5SH4mxNBWa/giphy.gif",
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_5":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Muy bien te comparto el canal de Youtube \n",
|
|
||||||
"https://youtube.com/leifermendez \n"
|
|
||||||
],
|
|
||||||
"media":null,
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_6":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Perfecto, te voy a pasar la lista ",
|
|
||||||
"de los temas que tengo y un breve video 🙂🤖 \n\n",
|
|
||||||
"*Angular* Basico (Pago) \n",
|
|
||||||
"*Angular* Basico (Gratis) \n",
|
|
||||||
"*Node* Basico (Gratis) \n",
|
|
||||||
"*NGRX* Basico (Gratis) \n",
|
|
||||||
"*AWS* Basico (Pago) \n\n",
|
|
||||||
"Escribe la palabra del tema que te interese \n"
|
|
||||||
],
|
|
||||||
"media":"https://i.giphy.com/media/5J5gN0WUk0VToHaK2p/giphy-downsized.gif",
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_7":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Vente al telegram \n",
|
|
||||||
"https://t.me/leifermendez \n"
|
|
||||||
],
|
|
||||||
"media":null,
|
|
||||||
"trigger":null
|
|
||||||
},
|
|
||||||
"STEP_8":{
|
|
||||||
"replyMessage":[
|
|
||||||
"Esto es una nota de voz \n"
|
|
||||||
],
|
|
||||||
"media":"nota-de-voz.mp3",
|
|
||||||
"trigger":null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Binary file not shown.
Binary file not shown.
@@ -1,21 +0,0 @@
|
|||||||
const middlewareClient = (client = null) => async (req, res, next) => {
|
|
||||||
try {
|
|
||||||
|
|
||||||
if(!client){
|
|
||||||
res.status(409)
|
|
||||||
console.log(client)
|
|
||||||
res.send({ error: 'Error de client.' })
|
|
||||||
}else{
|
|
||||||
req.clientWs = client;
|
|
||||||
next()
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
} catch (e) {
|
|
||||||
console.log(e)
|
|
||||||
res.status(409)
|
|
||||||
res.send({ error: 'Error de client' })
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
module.exports = { middlewareClient }
|
|
||||||
9380
package-lock.json
generated
9380
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
106
package.json
106
package.json
@@ -1,12 +1,51 @@
|
|||||||
{
|
{
|
||||||
"name": "bot-whatsapp",
|
"name": "@bot-whatsapp/root",
|
||||||
"version": "1.0.0",
|
"version": "0.1.21",
|
||||||
"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,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "node ./app.js",
|
"commit": "git-cz",
|
||||||
"test": "echo \"Error: no test specified\" && exit 1"
|
"cli:rollup": "rollup --config ./packages/cli/rollup-cli.config.js ",
|
||||||
|
"create-bot:rollup": "rollup --config ./packages/create-bot-whatsapp/rollup-create.config.js ",
|
||||||
|
"bot:rollup": "rollup --config ./packages/bot/rollup-bot.config.js",
|
||||||
|
"provider:rollup": "rollup --config ./packages/provider/rollup-provider.config.js ",
|
||||||
|
"contexts:rollup": "rollup --config ./packages/contexts/rollup-contexts.config.js",
|
||||||
|
"database:rollup": "rollup --config ./packages/database/rollup-database.config.js",
|
||||||
|
"create-bot-whatsapp:rollup": "rollup --config ./packages/create-bot-whatsapp/rollup-create.config.js",
|
||||||
|
"portal:rollup": "rollup --config ./packages/portal/rollup-portal.config.js",
|
||||||
|
"format:check": "prettier --check ./packages",
|
||||||
|
"format:write": "prettier --write ./packages",
|
||||||
|
"fmt.staged": "pretty-quick --staged",
|
||||||
|
"lint:check": "eslint ./packages",
|
||||||
|
"lint:fix": "eslint --fix ./packages",
|
||||||
|
"build:portal-web": "cd ./packages/portal/ && yarn run build.types && yarn run build.client && yarn run build.server && yarn run lint --fix",
|
||||||
|
"build:full": "yarn run build:portal-web && yarn run cli:rollup && yarn run bot:rollup && yarn run provider:rollup && yarn run database:rollup && yarn run contexts:rollup && yarn run create-bot-whatsapp:rollup && yarn run portal:rollup",
|
||||||
|
"build": "yarn run cli:rollup && yarn run bot:rollup && yarn run provider:rollup && yarn run database:rollup && yarn run contexts:rollup && yarn run create-bot-whatsapp:rollup && yarn run portal:rollup",
|
||||||
|
"copy.lib": "node ./scripts/move.js",
|
||||||
|
"test.unit": "node ./node_modules/uvu/bin.js packages test",
|
||||||
|
"test.e2e": "node ./node_modules/uvu/bin.js __test__",
|
||||||
|
"test.coverage": "node ./node_modules/c8/bin/c8.js npm run test.unit",
|
||||||
|
"test": "npm run test.coverage",
|
||||||
|
"cli": "node ./packages/cli/bin/cli.js",
|
||||||
|
"create": "node ./packages/create-bot-whatsapp/bin/create.js",
|
||||||
|
"dev:debug": "node --inspect ./example-app/app.js",
|
||||||
|
"dev": "node ./example-app/app.js",
|
||||||
|
"prepare": "npx husky install",
|
||||||
|
"preinstall": "npx only-allow yarn",
|
||||||
|
"postinstall": "npx prettier --write .",
|
||||||
|
"release": "standard-version -- --prerelease --global"
|
||||||
},
|
},
|
||||||
|
"workspaces": [
|
||||||
|
"packages/create-bot-whatsapp",
|
||||||
|
"packages/bot",
|
||||||
|
"packages/cli",
|
||||||
|
"packages/database",
|
||||||
|
"packages/provider",
|
||||||
|
"packages/contexts",
|
||||||
|
"packages/portal",
|
||||||
|
"packages/docs"
|
||||||
|
],
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"whatsapp",
|
"whatsapp",
|
||||||
"bot-whatsapp",
|
"bot-whatsapp",
|
||||||
@@ -24,33 +63,42 @@
|
|||||||
"url": "https://github.com/aurik3"
|
"url": "https://github.com/aurik3"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"repository": {
|
"repository": "https://github.com/leifermendez/bot-whatsapp",
|
||||||
"type": "git",
|
|
||||||
"url": "https://github.com/leifermendez/bot-whatsapp"
|
|
||||||
},
|
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"dependencies": {
|
|
||||||
"@google-cloud/dialogflow": "^5.2.0",
|
|
||||||
"cors": "^2.8.5",
|
|
||||||
"dotenv": "^16.0.1",
|
|
||||||
"exceljs": "^4.3.0",
|
|
||||||
"express": "^4.18.1",
|
|
||||||
"file-type": "^17.1.6",
|
|
||||||
"mime-db": "^1.52.0",
|
|
||||||
"moment": "^2.29.4",
|
|
||||||
"mysql": "^2.18.1",
|
|
||||||
"qr-image": "^3.2.0",
|
|
||||||
"qrcode-terminal": "^0.12.0",
|
|
||||||
"socket.io": "^4.5.1",
|
|
||||||
"stormdb": "^0.6.0",
|
|
||||||
"whatsapp-web.js": "^1.18.4",
|
|
||||||
"xlsx": "^0.18.5"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"pm2": "^5.2.0",
|
"@commitlint/cli": "^17.3.0",
|
||||||
"prettier": "2.7.1"
|
"@commitlint/config-conventional": "^17.3.0",
|
||||||
|
"@octokit/core": "^4.1.0",
|
||||||
|
"@rollup/plugin-commonjs": "^23.0.2",
|
||||||
|
"@rollup/plugin-json": "^5.0.1",
|
||||||
|
"@rollup/plugin-node-resolve": "^15.0.1",
|
||||||
|
"@rollup/plugin-replace": "^5.0.1",
|
||||||
|
"c8": "^7.12.0",
|
||||||
|
"conventional-changelog": "^3.1.25",
|
||||||
|
"cross-env": "^7.0.3",
|
||||||
|
"eslint": "^8.26.0",
|
||||||
|
"eslint-config-prettier": "^8.5.0",
|
||||||
|
"fs-extra": "^11.1.0",
|
||||||
|
"git-cz": "^4.9.0",
|
||||||
|
"husky": "^8.0.2",
|
||||||
|
"mime-types": "^2.1.35",
|
||||||
|
"only-allow": "^1.1.1",
|
||||||
|
"prettier": "^2.8.0",
|
||||||
|
"pretty-quick": "^3.1.3",
|
||||||
|
"prompts": "^2.4.2",
|
||||||
|
"rimraf": "^3.0.2",
|
||||||
|
"rollup": "^3.2.3",
|
||||||
|
"rollup-plugin-cleanup": "^3.2.1",
|
||||||
|
"rollup-plugin-copy": "^3.4.0",
|
||||||
|
"semver": "^7.3.8",
|
||||||
|
"standard-version": "^9.5.0",
|
||||||
|
"uvu": "^0.5.6"
|
||||||
},
|
},
|
||||||
|
"packageManager": "yarn@3.3.0",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "16.x"
|
"node": ">=16",
|
||||||
}
|
"npm": "please-use-yarn",
|
||||||
|
"yarn": ">=3"
|
||||||
|
},
|
||||||
|
"author": "Leifer Mendez <leifer33@gmail.com>"
|
||||||
}
|
}
|
||||||
|
|||||||
110
packages/bot/USES_CASES.md
Normal file
110
packages/bot/USES_CASES.md
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|
# @bot-whatsapp/io
|
||||||
|
|
||||||
|
### Caso de uso
|
||||||
|
|
||||||
|
> Una persona escribe `hola`
|
||||||
|
|
||||||
|
**addKeyword** recibe `string | string[]`
|
||||||
|
|
||||||
|
> `sensitive` false _default_
|
||||||
|
|
||||||
|
- [x] addKeyword
|
||||||
|
- [x] addAnswer
|
||||||
|
- [x] addKeyword: Opciones
|
||||||
|
- [x] addAnswer: Opciones, media, buttons
|
||||||
|
- [x] Retornar JSON (options)
|
||||||
|
- [ ] Recibir JSON
|
||||||
|
|
||||||
|
```js
|
||||||
|
// bootstrap.js Como iniciar el provider
|
||||||
|
const { inout, provider, database } = require('@bot-whatsapp')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* async whatsapp-web, twilio, meta
|
||||||
|
* */
|
||||||
|
|
||||||
|
const bootstrap = async () => {
|
||||||
|
console.log(`Iniciando....`)
|
||||||
|
const client = await provider.start()
|
||||||
|
/**
|
||||||
|
* - QR
|
||||||
|
* - Endpoint
|
||||||
|
* - Check Token Meta, Twilio
|
||||||
|
* - Return events? on message
|
||||||
|
* */
|
||||||
|
console.log(`Fin...`)
|
||||||
|
// Esto es opcional ? no deberia ser necesario
|
||||||
|
client.on('message', ({number, body,...}) => {
|
||||||
|
// Incoming message
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
// flow.js Como agregar keywords y respuestas
|
||||||
|
const { inout, provider, database } = require('@bot-whatsapp')
|
||||||
|
|
||||||
|
await inout
|
||||||
|
.addKeyword('hola')
|
||||||
|
.addAnswer('Bienvenido a tu tienda 🥲')
|
||||||
|
.addAnswer('escribe *catalogo* o *ofertas*')
|
||||||
|
|
||||||
|
await inout
|
||||||
|
.addKeyword(['catalogo', 'ofertas'])
|
||||||
|
.addAnswer('Este es nuestro CATALOGO mas reciente!', {
|
||||||
|
buttons: [{ body: 'Xiaomi' }, { body: 'Samsung' }],
|
||||||
|
})
|
||||||
|
|
||||||
|
await inout
|
||||||
|
.addKeyword('Xiaomi')
|
||||||
|
.addAnswer('Estos son nuestro productos XIAOMI ....', {
|
||||||
|
media: 'https://....',
|
||||||
|
})
|
||||||
|
.addAnswer('Si quieres mas info escrbie *info*')
|
||||||
|
|
||||||
|
await inout
|
||||||
|
.addKeyword('chao!')
|
||||||
|
.addAnswer('bye!')
|
||||||
|
.addAnswer('Recuerda que tengo esta promo', {
|
||||||
|
media: 'https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif',
|
||||||
|
})
|
||||||
|
|
||||||
|
await inout
|
||||||
|
.addKeyword('Modelo C', { sensitive: false })
|
||||||
|
.addAnswer('100USD', { media: 'http//:...' })
|
||||||
|
|
||||||
|
await inout
|
||||||
|
.addKeyword('hola!', { sensitive: false })
|
||||||
|
.addAnswer('Bievenido Escribe *productos*')
|
||||||
|
|
||||||
|
await inout
|
||||||
|
.addKeyword('productos', { sensitive: false })
|
||||||
|
.addAnswer('Esto son los mas vendidos')
|
||||||
|
.addAnswer('*PC1* Precio 10USD', { media: 'https://....' })
|
||||||
|
.addAnswer('*PC2* Precio 10USD', { media: 'https://....' })
|
||||||
|
|
||||||
|
await inout
|
||||||
|
.addKeyword('PC1', { sensitive: false })
|
||||||
|
.addAnswer('Bievenido Escribe *productos*')
|
||||||
|
|
||||||
|
const answerOne = await inout.addAnswer({
|
||||||
|
message: 'Como estas!',
|
||||||
|
media: 'https://media2.giphy.com/media/VQJu0IeULuAmCwf5SL/giphy.gif',
|
||||||
|
})
|
||||||
|
|
||||||
|
const otherAnswer = await inout.addAnswer('Aprovecho para decirte!')
|
||||||
|
|
||||||
|
answerOne.push(otherAnswer)
|
||||||
|
|
||||||
|
inout.addKeywords(['hola', 'hi', 'ola'])
|
||||||
|
```
|
||||||
|
|
||||||
|
**Comunidad**
|
||||||
|
|
||||||
|
> Forma parte de este proyecto.
|
||||||
|
|
||||||
|
- [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)
|
||||||
300
packages/bot/core/core.class.js
Normal file
300
packages/bot/core/core.class.js
Normal file
@@ -0,0 +1,300 @@
|
|||||||
|
const { toCtx } = require('../io/methods')
|
||||||
|
const { printer } = require('../utils/interactive')
|
||||||
|
const { delay } = require('../utils/delay')
|
||||||
|
const Queue = require('../utils/queue')
|
||||||
|
const { Console } = require('console')
|
||||||
|
const { createWriteStream } = require('fs')
|
||||||
|
|
||||||
|
const logger = new Console({
|
||||||
|
stdout: createWriteStream(`${process.cwd()}/core.class.log`),
|
||||||
|
})
|
||||||
|
|
||||||
|
const QueuePrincipal = new Queue()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* [ ] Escuchar eventos del provider asegurarte que los provider emitan eventos
|
||||||
|
* [ ] Guardar historial en db
|
||||||
|
* [ ] Buscar mensaje en flow
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
class CoreClass {
|
||||||
|
flowClass
|
||||||
|
databaseClass
|
||||||
|
providerClass
|
||||||
|
generalArgs = { blackList: [] }
|
||||||
|
constructor(_flow, _database, _provider, _args) {
|
||||||
|
this.flowClass = _flow
|
||||||
|
this.databaseClass = _database
|
||||||
|
this.providerClass = _provider
|
||||||
|
this.generalArgs = { ...this.generalArgs, ..._args }
|
||||||
|
|
||||||
|
for (const { event, func } of this.listenerBusEvents()) {
|
||||||
|
this.providerClass.on(event, func)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Manejador de eventos
|
||||||
|
*/
|
||||||
|
listenerBusEvents = () => [
|
||||||
|
{
|
||||||
|
event: 'preinit',
|
||||||
|
func: () => printer('Iniciando proveedor, espere...'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
event: 'require_action',
|
||||||
|
func: ({ instructions, title = '⚡⚡ ACCIÓN REQUERIDA ⚡⚡' }) => printer(instructions, title),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
event: 'ready',
|
||||||
|
func: () => printer('Proveedor conectado y listo'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
event: 'auth_failure',
|
||||||
|
func: ({ instructions }) => printer(instructions, '⚡⚡ ERROR AUTH ⚡⚡'),
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
event: 'message',
|
||||||
|
func: (msg) => this.handleMsg(msg),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GLOSSARY.md
|
||||||
|
* @param {*} messageCtxInComming
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
handleMsg = async (messageCtxInComming) => {
|
||||||
|
logger.log(`[handleMsg]: `, messageCtxInComming)
|
||||||
|
const { body, from } = messageCtxInComming
|
||||||
|
let msgToSend = []
|
||||||
|
let endFlowFlag = false
|
||||||
|
let fallBackFlag = false
|
||||||
|
if (this.generalArgs.blackList.includes(from)) return
|
||||||
|
if (!body) return
|
||||||
|
if (!body.length) return
|
||||||
|
|
||||||
|
let prevMsg = await this.databaseClass.getPrevByNumber(from)
|
||||||
|
const refToContinue = this.flowClass.findBySerialize(prevMsg?.refSerialize)
|
||||||
|
|
||||||
|
if (prevMsg?.ref) {
|
||||||
|
const ctxByNumber = toCtx({
|
||||||
|
body,
|
||||||
|
from,
|
||||||
|
prevRef: prevMsg.refSerialize,
|
||||||
|
})
|
||||||
|
await this.databaseClass.save(ctxByNumber)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄 Crar CTX de mensaje (uso private)
|
||||||
|
const createCtxMessage = (payload = {}, index = 0) => {
|
||||||
|
const body = typeof payload === 'string' ? payload : payload?.body ?? payload?.answer
|
||||||
|
const media = payload?.media ?? null
|
||||||
|
const buttons = payload?.buttons ?? []
|
||||||
|
const capture = payload?.capture ?? false
|
||||||
|
|
||||||
|
return toCtx({
|
||||||
|
body,
|
||||||
|
from,
|
||||||
|
keyword: null,
|
||||||
|
index,
|
||||||
|
options: { media, buttons, capture },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄 Limpiar cola de procesos
|
||||||
|
const clearQueue = () => {
|
||||||
|
QueuePrincipal.pendingPromise = false
|
||||||
|
QueuePrincipal.queue = []
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄 Finalizar flujo
|
||||||
|
const endFlow =
|
||||||
|
(flag) =>
|
||||||
|
async (message = null) => {
|
||||||
|
flag.endFlow = true
|
||||||
|
endFlowFlag = true
|
||||||
|
if (message) this.sendProviderAndSave(from, createCtxMessage(message))
|
||||||
|
clearQueue()
|
||||||
|
sendFlow([])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄 Esta funcion se encarga de enviar un array de mensajes dentro de este ctx
|
||||||
|
const sendFlow = async (messageToSend, numberOrId, options = { prev: prevMsg }) => {
|
||||||
|
if (options.prev?.options?.capture) await cbEveryCtx(options.prev?.ref)
|
||||||
|
|
||||||
|
const queue = []
|
||||||
|
for (const ctxMessage of messageToSend) {
|
||||||
|
if (endFlowFlag) return
|
||||||
|
const delayMs = ctxMessage?.options?.delay || 0
|
||||||
|
if (delayMs) await delay(delayMs)
|
||||||
|
await QueuePrincipal.enqueue(() =>
|
||||||
|
this.sendProviderAndSave(numberOrId, ctxMessage).then(() => resolveCbEveryCtx(ctxMessage))
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return Promise.all(queue)
|
||||||
|
}
|
||||||
|
|
||||||
|
const continueFlow = async () => {
|
||||||
|
const currentPrev = await this.databaseClass.getPrevByNumber(from)
|
||||||
|
const nextFlow = (await this.flowClass.find(refToContinue?.ref, true)) ?? []
|
||||||
|
const filterNextFlow = nextFlow.filter((msg) => msg.refSerialize !== currentPrev?.refSerialize)
|
||||||
|
const isContinueFlow = filterNextFlow.map((i) => i.keyword).includes(currentPrev?.ref)
|
||||||
|
|
||||||
|
if (!isContinueFlow) {
|
||||||
|
const refToContinueChild = this.flowClass.getRefToContinueChild(currentPrev?.keyword)
|
||||||
|
const flowStandaloneChild = this.flowClass.getFlowsChild()
|
||||||
|
const nextChildMessages =
|
||||||
|
(await this.flowClass.find(refToContinueChild?.ref, true, flowStandaloneChild)) || []
|
||||||
|
if (nextChildMessages?.length) return await sendFlow(nextChildMessages, from, { prev: undefined })
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isContinueFlow) {
|
||||||
|
await sendFlow(filterNextFlow, from, { prev: undefined })
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 📄 [options: fallBack]: esta funcion se encarga de repetir el ultimo mensaje
|
||||||
|
const fallBack =
|
||||||
|
(flag) =>
|
||||||
|
async (message = null) => {
|
||||||
|
QueuePrincipal.queue = []
|
||||||
|
flag.fallBack = true
|
||||||
|
await this.sendProviderAndSave(from, {
|
||||||
|
...prevMsg,
|
||||||
|
answer: typeof message === 'string' ? message : message?.body ?? prevMsg.answer,
|
||||||
|
options: {
|
||||||
|
...prevMsg.options,
|
||||||
|
buttons: prevMsg.options?.buttons,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄 [options: flowDynamic]: esta funcion se encarga de responder un array de respuesta esta limitado a 5 mensajes
|
||||||
|
// para evitar bloque de whatsapp
|
||||||
|
|
||||||
|
const flowDynamic =
|
||||||
|
(flag) =>
|
||||||
|
async (listMsg = []) => {
|
||||||
|
flag.flowDynamic = true
|
||||||
|
if (!Array.isArray(listMsg)) listMsg = [listMsg]
|
||||||
|
|
||||||
|
const parseListMsg = listMsg.map((opt, index) => createCtxMessage(opt, index))
|
||||||
|
|
||||||
|
if (endFlowFlag) return
|
||||||
|
for (const msg of parseListMsg) {
|
||||||
|
await this.sendProviderAndSave(from, msg)
|
||||||
|
}
|
||||||
|
await continueFlow()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄 Se encarga de revisar si el contexto del mensaje tiene callback o fallback
|
||||||
|
const resolveCbEveryCtx = async (ctxMessage) => {
|
||||||
|
if (!ctxMessage?.options?.capture) return await cbEveryCtx(ctxMessage?.ref)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄 Se encarga de revisar si el contexto del mensaje tiene callback y ejecutarlo
|
||||||
|
const cbEveryCtx = async (inRef) => {
|
||||||
|
let flags = {
|
||||||
|
endFlow: false,
|
||||||
|
fallBack: false,
|
||||||
|
flowDynamic: false,
|
||||||
|
wait: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
const provider = this.providerClass
|
||||||
|
|
||||||
|
if (!this.flowClass.allCallbacks[inRef]) return Promise.resolve()
|
||||||
|
|
||||||
|
const argsCb = {
|
||||||
|
provider,
|
||||||
|
fallBack: fallBack(flags),
|
||||||
|
flowDynamic: flowDynamic(flags),
|
||||||
|
endFlow: endFlow(flags),
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.flowClass.allCallbacks[inRef](messageCtxInComming, argsCb)
|
||||||
|
const wait = !(!flags.endFlow && !flags.fallBack && !flags.flowDynamic)
|
||||||
|
if (!wait) await continueFlow()
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄🤘(tiene return) [options: nested(array)]: Si se tiene flujos hijos los implementa
|
||||||
|
if (!endFlowFlag && prevMsg?.options?.nested?.length) {
|
||||||
|
const nestedRef = prevMsg.options.nested
|
||||||
|
const flowStandalone = nestedRef.map((f) => ({
|
||||||
|
...nestedRef.find((r) => r.refSerialize === f.refSerialize),
|
||||||
|
}))
|
||||||
|
|
||||||
|
msgToSend = this.flowClass.find(body, false, flowStandalone) || []
|
||||||
|
|
||||||
|
await sendFlow(msgToSend, from)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 📄🤘(tiene return) Si el mensaje previo implementa capture
|
||||||
|
if (!endFlowFlag && !prevMsg?.options?.nested?.length) {
|
||||||
|
const typeCapture = typeof prevMsg?.options?.capture
|
||||||
|
|
||||||
|
if (typeCapture === 'boolean' && fallBackFlag) {
|
||||||
|
msgToSend = this.flowClass.find(refToContinue?.ref, true) || []
|
||||||
|
await sendFlow(msgToSend, from)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
msgToSend = this.flowClass.find(body) || []
|
||||||
|
sendFlow(msgToSend, from)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enviar mensaje con contexto atraves del proveedor de whatsapp
|
||||||
|
* @param {*} numberOrId
|
||||||
|
* @param {*} ctxMessage ver más en GLOSSARY.md
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
sendProviderAndSave = async (numberOrId, ctxMessage) => {
|
||||||
|
const { answer } = ctxMessage
|
||||||
|
await this.providerClass.sendMessage(numberOrId, answer, ctxMessage)
|
||||||
|
await this.databaseClass.save({ ...ctxMessage, from: numberOrId })
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated
|
||||||
|
* @private
|
||||||
|
* @param {*} message
|
||||||
|
* @param {*} ref
|
||||||
|
*/
|
||||||
|
continue = (message, ref = false) => {
|
||||||
|
const responde = this.flowClass.find(message, ref)
|
||||||
|
if (responde) {
|
||||||
|
this.providerClass.sendMessage(responde.answer)
|
||||||
|
this.databaseClass.saveLog(responde.answer)
|
||||||
|
this.continue(null, responde.ref)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Funcion dedicada a enviar el mensaje sin pasar por el flow
|
||||||
|
* (dialogflow)
|
||||||
|
* @param {*} messageToSend
|
||||||
|
* @param {*} numberOrId
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
sendFlowSimple = async (messageToSend, numberOrId) => {
|
||||||
|
const queue = []
|
||||||
|
for (const ctxMessage of messageToSend) {
|
||||||
|
const delayMs = ctxMessage?.options?.delay || 0
|
||||||
|
if (delayMs) await delay(delayMs)
|
||||||
|
QueuePrincipal.enqueue(() => this.sendProviderAndSave(numberOrId, ctxMessage))
|
||||||
|
}
|
||||||
|
return Promise.all(queue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
module.exports = CoreClass
|
||||||
45
packages/bot/index.js
Normal file
45
packages/bot/index.js
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
const CoreClass = require('./core/core.class')
|
||||||
|
const ProviderClass = require('./provider/provider.class')
|
||||||
|
const FlowClass = require('./io/flow.class')
|
||||||
|
const { addKeyword, addAnswer, addChild, toSerialize } = require('./io/methods')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Crear instancia de clase Bot
|
||||||
|
* @param {*} args
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const createBot = async ({ flow, database, provider }, args = {}) => new CoreClass(flow, database, provider, args)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Crear instancia de clase Io (Flow)
|
||||||
|
* @param {*} args
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const createFlow = (args) => {
|
||||||
|
return new FlowClass(args)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Crear instancia de clase Provider
|
||||||
|
* Depdendiendo del Provider puedes pasar argumentos
|
||||||
|
* Ver Documentacion
|
||||||
|
* @param {*} args
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const createProvider = (providerClass = class {}, args = null) => {
|
||||||
|
const providerInstance = new providerClass(args)
|
||||||
|
if (!providerClass.prototype instanceof ProviderClass) throw new Error('El provider no implementa ProviderClass')
|
||||||
|
return providerInstance
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
createBot,
|
||||||
|
createFlow,
|
||||||
|
createProvider,
|
||||||
|
addKeyword,
|
||||||
|
addAnswer,
|
||||||
|
addChild,
|
||||||
|
toSerialize,
|
||||||
|
ProviderClass,
|
||||||
|
CoreClass,
|
||||||
|
}
|
||||||
95
packages/bot/io/flow.class.js
Normal file
95
packages/bot/io/flow.class.js
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
const { toSerialize } = require('./methods/toSerialize')
|
||||||
|
const { flatObject } = require('../utils/flattener')
|
||||||
|
|
||||||
|
class FlowClass {
|
||||||
|
allCallbacks = []
|
||||||
|
flowSerialize = []
|
||||||
|
flowRaw = []
|
||||||
|
constructor(_flow) {
|
||||||
|
if (!Array.isArray(_flow)) throw new Error('Esto debe ser un ARRAY')
|
||||||
|
this.flowRaw = _flow
|
||||||
|
|
||||||
|
this.allCallbacks = flatObject(_flow)
|
||||||
|
|
||||||
|
const mergeToJsonSerialize = Object.keys(_flow)
|
||||||
|
.map((indexObjectFlow) => _flow[indexObjectFlow].toJson())
|
||||||
|
.flat(2)
|
||||||
|
|
||||||
|
this.flowSerialize = toSerialize(mergeToJsonSerialize)
|
||||||
|
}
|
||||||
|
|
||||||
|
find = (keyOrWord, symbol = false, overFlow = null) => {
|
||||||
|
keyOrWord = `${keyOrWord}`
|
||||||
|
let capture = false
|
||||||
|
let messages = []
|
||||||
|
let refSymbol = null
|
||||||
|
overFlow = overFlow ?? this.flowSerialize
|
||||||
|
|
||||||
|
const mapSensitive = (str, mapOptions = { sensitive: false, regex: false }) => {
|
||||||
|
if (mapOptions.regex) return new RegExp(str)
|
||||||
|
const regexSensitive = mapOptions.sensitive ? 'g' : 'i'
|
||||||
|
if (Array.isArray(str)) {
|
||||||
|
return new RegExp(str.join('|'), regexSensitive)
|
||||||
|
}
|
||||||
|
return new RegExp(str, regexSensitive)
|
||||||
|
}
|
||||||
|
|
||||||
|
const findIn = (keyOrWord, symbol = false, flow = overFlow) => {
|
||||||
|
capture = refSymbol?.options?.capture || false
|
||||||
|
if (capture) return messages
|
||||||
|
|
||||||
|
if (symbol) {
|
||||||
|
refSymbol = flow.find((c) => c.keyword === keyOrWord)
|
||||||
|
if (refSymbol?.answer) messages.push(refSymbol)
|
||||||
|
if (refSymbol?.ref) findIn(refSymbol.ref, true)
|
||||||
|
} else {
|
||||||
|
refSymbol = flow.find((c) => {
|
||||||
|
const sensitive = c?.options?.sensitive || false
|
||||||
|
const regex = c?.options?.regex || false
|
||||||
|
return mapSensitive(c.keyword, { sensitive, regex }).test(keyOrWord)
|
||||||
|
})
|
||||||
|
if (refSymbol?.ref) findIn(refSymbol.ref, true)
|
||||||
|
return messages
|
||||||
|
}
|
||||||
|
}
|
||||||
|
findIn(keyOrWord, symbol)
|
||||||
|
return messages
|
||||||
|
}
|
||||||
|
|
||||||
|
findBySerialize = (refSerialize) => this.flowSerialize.find((r) => r.refSerialize === refSerialize)
|
||||||
|
|
||||||
|
findIndexByRef = (ref) => this.flowSerialize.findIndex((r) => r.ref === ref)
|
||||||
|
|
||||||
|
getRefToContinueChild = (keyword) => {
|
||||||
|
try {
|
||||||
|
const flowChilds = this.flowSerialize
|
||||||
|
.reduce((acc, cur) => {
|
||||||
|
const merge = [...acc, cur?.options?.nested].flat(2)
|
||||||
|
return merge
|
||||||
|
}, [])
|
||||||
|
.filter((i) => !!i && i?.refSerialize === keyword)
|
||||||
|
.shift()
|
||||||
|
|
||||||
|
return flowChilds
|
||||||
|
} catch (e) {
|
||||||
|
return undefined
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getFlowsChild = () => {
|
||||||
|
try {
|
||||||
|
const flowChilds = this.flowSerialize
|
||||||
|
.reduce((acc, cur) => {
|
||||||
|
const merge = [...acc, cur?.options?.nested].flat(2)
|
||||||
|
return merge
|
||||||
|
}, [])
|
||||||
|
.filter((i) => !!i)
|
||||||
|
|
||||||
|
return flowChilds
|
||||||
|
} catch (e) {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = FlowClass
|
||||||
105
packages/bot/io/methods/addAnswer.js
Normal file
105
packages/bot/io/methods/addAnswer.js
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
const { flatObject } = require('../../utils/flattener')
|
||||||
|
const { generateRef } = require('../../utils/hash')
|
||||||
|
const { addChild } = require('./addChild')
|
||||||
|
const { toJson } = require('./toJson')
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param answer string
|
||||||
|
* @param options {media:string, buttons:[{"body":"😎 Cursos"}], delay:ms, capture:true default false}
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const addAnswer =
|
||||||
|
(inCtx) =>
|
||||||
|
(answer, options, cb = null, nested = []) => {
|
||||||
|
answer = Array.isArray(answer) ? answer.join('\n') : answer
|
||||||
|
/**
|
||||||
|
* Todas las opciones referentes a el mensaje en concreto options:{}
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const getAnswerOptions = () => ({
|
||||||
|
media: typeof options?.media === 'string' ? `${options?.media}` : null,
|
||||||
|
buttons: Array.isArray(options?.buttons) ? options.buttons : [],
|
||||||
|
capture: typeof options?.capture === 'boolean' ? options?.capture : false,
|
||||||
|
child: typeof options?.child === 'string' ? `${options?.child}` : null,
|
||||||
|
delay: typeof options?.delay === 'number' ? options?.delay : 0,
|
||||||
|
})
|
||||||
|
|
||||||
|
const getNested = () => {
|
||||||
|
let flatNested = []
|
||||||
|
if (Array.isArray(nested)) {
|
||||||
|
for (const iterator of nested) {
|
||||||
|
flatNested = [...flatNested, ...addChild(iterator)]
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
nested: flatNested,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
nested: addChild(nested),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Esta funcion aplana y busca los callback anidados de los hijos
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const getCbFromNested = () => flatObject(Array.isArray(nested) ? nested : [nested])
|
||||||
|
|
||||||
|
const callback = typeof cb === 'function' ? cb : () => null
|
||||||
|
|
||||||
|
const lastCtx = inCtx.hasOwnProperty('ctx') ? inCtx.ctx : inCtx
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Esta funcion se encarga de mapear y transformar todo antes
|
||||||
|
* de retornar
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const ctxAnswer = () => {
|
||||||
|
const ref = `ans_${generateRef()}`
|
||||||
|
|
||||||
|
const options = {
|
||||||
|
...getAnswerOptions(),
|
||||||
|
...getNested(),
|
||||||
|
keyword: {},
|
||||||
|
callback: !!cb,
|
||||||
|
}
|
||||||
|
|
||||||
|
const json = [].concat(inCtx.json).concat([
|
||||||
|
{
|
||||||
|
ref,
|
||||||
|
keyword: lastCtx.ref,
|
||||||
|
answer,
|
||||||
|
options,
|
||||||
|
},
|
||||||
|
])
|
||||||
|
|
||||||
|
getCbFromNested()
|
||||||
|
const callbacks = {
|
||||||
|
...inCtx.callbacks,
|
||||||
|
...getCbFromNested(),
|
||||||
|
[ref]: callback,
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...lastCtx,
|
||||||
|
ref,
|
||||||
|
answer,
|
||||||
|
json,
|
||||||
|
options,
|
||||||
|
callbacks,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Retornar contexto no colocar nada más abajo de esto
|
||||||
|
const ctx = ctxAnswer()
|
||||||
|
|
||||||
|
return {
|
||||||
|
ctx,
|
||||||
|
ref: ctx.ref,
|
||||||
|
addAnswer: addAnswer(ctx),
|
||||||
|
toJson: toJson(ctx),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { addAnswer }
|
||||||
15
packages/bot/io/methods/addChild.js
Normal file
15
packages/bot/io/methods/addChild.js
Normal 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 }
|
||||||
51
packages/bot/io/methods/addKeyword.js
Normal file
51
packages/bot/io/methods/addKeyword.js
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
const { generateRef } = require('../../utils/hash')
|
||||||
|
const { addAnswer } = require('./addAnswer')
|
||||||
|
const { toJson } = require('./toJson')
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {*} message `string | string[]`
|
||||||
|
* @param {*} options {sensitive:boolean} default false
|
||||||
|
*/
|
||||||
|
const addKeyword = (keyword, options) => {
|
||||||
|
if (typeof keyword !== 'string' && !Array.isArray(keyword)) {
|
||||||
|
throw new Error('DEBE_SER_STRING_ARRAY_REGEX')
|
||||||
|
}
|
||||||
|
|
||||||
|
const parseOptions = () => {
|
||||||
|
const defaultProperties = {
|
||||||
|
sensitive: typeof options?.sensitive === 'boolean' ? options?.sensitive : false,
|
||||||
|
regex: typeof options?.regex === 'boolean' ? options?.regex : false,
|
||||||
|
}
|
||||||
|
|
||||||
|
return defaultProperties
|
||||||
|
}
|
||||||
|
|
||||||
|
const ctxAddKeyword = () => {
|
||||||
|
const ref = `key_${generateRef()}`
|
||||||
|
const options = parseOptions()
|
||||||
|
const json = [
|
||||||
|
{
|
||||||
|
ref,
|
||||||
|
keyword,
|
||||||
|
options,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
/**
|
||||||
|
* Se guarda en db
|
||||||
|
*/
|
||||||
|
|
||||||
|
return { ref, keyword, options, json }
|
||||||
|
}
|
||||||
|
|
||||||
|
const ctx = ctxAddKeyword()
|
||||||
|
|
||||||
|
return {
|
||||||
|
ctx,
|
||||||
|
ref: ctx.ref,
|
||||||
|
addAnswer: addAnswer(ctx),
|
||||||
|
toJson: toJson(ctx),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { addKeyword }
|
||||||
8
packages/bot/io/methods/index.js
Normal file
8
packages/bot/io/methods/index.js
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
const { addAnswer } = require('./addAnswer')
|
||||||
|
const { addKeyword } = require('./addKeyword')
|
||||||
|
const { addChild } = require('./addChild')
|
||||||
|
const { toSerialize } = require('./toSerialize')
|
||||||
|
const { toCtx } = require('./toCtx')
|
||||||
|
const { toJson } = require('./toJson')
|
||||||
|
|
||||||
|
module.exports = { addAnswer, addKeyword, addChild, toCtx, toJson, toSerialize }
|
||||||
19
packages/bot/io/methods/toCtx.js
Normal file
19
packages/bot/io/methods/toCtx.js
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
const { generateRef, generateRefSerialize } = require('../../utils/hash')
|
||||||
|
/**
|
||||||
|
* @deprecate
|
||||||
|
* @param answer string
|
||||||
|
* @param options {media:string, buttons:[], capture:true default false}
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const toCtx = ({ body, from, prevRef, options = {}, index }) => {
|
||||||
|
return {
|
||||||
|
ref: generateRef(),
|
||||||
|
keyword: prevRef,
|
||||||
|
answer: body,
|
||||||
|
options: options ?? {},
|
||||||
|
from,
|
||||||
|
refSerialize: generateRefSerialize({ index, answer: body }),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { toCtx }
|
||||||
6
packages/bot/io/methods/toJson.js
Normal file
6
packages/bot/io/methods/toJson.js
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
const toJson = (inCtx) => () => {
|
||||||
|
const lastCtx = inCtx.hasOwnProperty('ctx') ? inCtx.ctx : inCtx
|
||||||
|
return lastCtx.json
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { toJson }
|
||||||
23
packages/bot/io/methods/toSerialize.js
Normal file
23
packages/bot/io/methods/toSerialize.js
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
const { generateRefSerialize } = require('../../utils/hash')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Crear referencia serializada
|
||||||
|
* @param {*} flowJson
|
||||||
|
* @returns array[]
|
||||||
|
*/
|
||||||
|
const toSerialize = (flowJson) => {
|
||||||
|
if (!Array.isArray(flowJson)) throw new Error('Esto debe ser un ARRAY')
|
||||||
|
|
||||||
|
const jsonToSerialize = flowJson.map((row, index) => ({
|
||||||
|
...row,
|
||||||
|
refSerialize: `${generateRefSerialize({
|
||||||
|
index,
|
||||||
|
keyword: row.keyword,
|
||||||
|
answer: row.answer,
|
||||||
|
})}`,
|
||||||
|
}))
|
||||||
|
|
||||||
|
return jsonToSerialize
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { toSerialize }
|
||||||
14
packages/bot/io/rollup-cli.config.js
Normal file
14
packages/bot/io/rollup-cli.config.js
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
const commonjs = require('@rollup/plugin-commonjs')
|
||||||
|
const { nodeResolve } = require('@rollup/plugin-node-resolve')
|
||||||
|
const { join } = require('path')
|
||||||
|
|
||||||
|
const PATH = join(__dirname, 'lib', 'io', 'bundle.io.cjs')
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
input: 'index.js',
|
||||||
|
output: {
|
||||||
|
file: PATH,
|
||||||
|
format: 'cjs',
|
||||||
|
},
|
||||||
|
plugins: [commonjs(), nodeResolve()],
|
||||||
|
}
|
||||||
36
packages/bot/package.json
Normal file
36
packages/bot/package.json
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
{
|
||||||
|
"name": "@bot-whatsapp/bot",
|
||||||
|
"version": "0.0.100-alpha.0",
|
||||||
|
"description": "",
|
||||||
|
"main": "./lib/bundle.bot.cjs",
|
||||||
|
"scripts": {
|
||||||
|
"bot:rollup": "node ../../node_modules/.bin/rollup index.js --config ./rollup-cli.config.js",
|
||||||
|
"format:check": "prettier --check .",
|
||||||
|
"format:write": "prettier --write .",
|
||||||
|
"lint:check": "eslint .",
|
||||||
|
"lint:fix": "eslint --fix .",
|
||||||
|
"test.unit": "cross-env NODE_ENV=test node ../../node_modules/uvu/bin.js tests"
|
||||||
|
},
|
||||||
|
"keywords": [],
|
||||||
|
"files": [
|
||||||
|
"./lib/bundle.bot.cjs",
|
||||||
|
"./provider/*",
|
||||||
|
"./core/*",
|
||||||
|
"./io/*"
|
||||||
|
],
|
||||||
|
"author": "",
|
||||||
|
"license": "ISC",
|
||||||
|
"devDependencies": {
|
||||||
|
"@bot-whatsapp/cli": "*",
|
||||||
|
"@bot-whatsapp/database": "*",
|
||||||
|
"@bot-whatsapp/provider": "*",
|
||||||
|
"kleur": "^4.1.5"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"dotenv": "^16.0.3"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/codigoencasa/bot-whatsapp/tree/main/packages/bot"
|
||||||
|
}
|
||||||
|
}
|
||||||
30
packages/bot/provider/provider.class.js
Normal file
30
packages/bot/provider/provider.class.js
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
const { EventEmitter } = require('node:events')
|
||||||
|
/**
|
||||||
|
* Esta clase debe siempre proporcionar los siguietes metodos
|
||||||
|
* sendMessage = Para enviar un mensaje
|
||||||
|
*
|
||||||
|
* @important
|
||||||
|
* Esta clase extiende de la clase del provider OJO
|
||||||
|
* Eventos
|
||||||
|
* - message
|
||||||
|
* - ready
|
||||||
|
* - error
|
||||||
|
* - require_action
|
||||||
|
*/
|
||||||
|
|
||||||
|
const NODE_ENV = process.env.NODE_ENV || 'dev'
|
||||||
|
class ProviderClass extends EventEmitter {
|
||||||
|
/**
|
||||||
|
* events: message | auth | auth_error | ...
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
sendMessage = async (userId, message) => {
|
||||||
|
if (NODE_ENV !== 'production') console.log('[sendMessage]', { userId, message })
|
||||||
|
return message
|
||||||
|
}
|
||||||
|
|
||||||
|
getInstance = () => this.vendor
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = ProviderClass
|
||||||
26
packages/bot/rollup-bot.config.js
Normal file
26
packages/bot/rollup-bot.config.js
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
const banner = require('../../config/banner.rollup.json')
|
||||||
|
const commonjs = require('@rollup/plugin-commonjs')
|
||||||
|
const { nodeResolve } = require('@rollup/plugin-node-resolve')
|
||||||
|
const { join } = require('path')
|
||||||
|
|
||||||
|
module.exports = [
|
||||||
|
{
|
||||||
|
input: join(__dirname, 'index.js'),
|
||||||
|
output: {
|
||||||
|
banner: banner['banner.output'].join(''),
|
||||||
|
file: join(__dirname, 'lib', 'bundle.bot.cjs'),
|
||||||
|
format: 'cjs',
|
||||||
|
sourcemap: true,
|
||||||
|
},
|
||||||
|
plugins: [commonjs(), nodeResolve()],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: join(__dirname, 'index.js'),
|
||||||
|
output: {
|
||||||
|
banner: banner['banner.output'].join(''),
|
||||||
|
file: join(__dirname, 'lib', 'bundle.bot.cjs'),
|
||||||
|
format: 'cjs',
|
||||||
|
},
|
||||||
|
plugins: [commonjs(), nodeResolve()],
|
||||||
|
},
|
||||||
|
]
|
||||||
227
packages/bot/tests/bot.class.test.js
Normal file
227
packages/bot/tests/bot.class.test.js
Normal file
@@ -0,0 +1,227 @@
|
|||||||
|
const { test } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const FlowClass = require('../io/flow.class')
|
||||||
|
const MockProvider = require('../../../__mocks__/mock.provider')
|
||||||
|
const { createBot, CoreClass, createFlow, createProvider, ProviderClass } = require('../index')
|
||||||
|
|
||||||
|
class MockFlow {
|
||||||
|
allCallbacks = { ref: () => 1 }
|
||||||
|
flowSerialize = []
|
||||||
|
flowRaw = []
|
||||||
|
find = (arg) => {
|
||||||
|
if (arg) {
|
||||||
|
return [{ answer: 'answer', ref: 'ref' }]
|
||||||
|
} else {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
findBySerialize = () => ({})
|
||||||
|
findIndexByRef = () => 0
|
||||||
|
getRefToContinueChild = () => ({})
|
||||||
|
getFlowsChild = () => []
|
||||||
|
}
|
||||||
|
|
||||||
|
class MockDBA {
|
||||||
|
listHistory = []
|
||||||
|
save = () => {}
|
||||||
|
getPrevByNumber = () => {}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MockDBB {
|
||||||
|
listHistory = []
|
||||||
|
save = () => {}
|
||||||
|
getPrevByNumber = () => ({
|
||||||
|
refSerialize: 'xxxxx',
|
||||||
|
ref: 'xxxx',
|
||||||
|
options: { callback: true },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
class MockDBC {
|
||||||
|
listHistory = []
|
||||||
|
save = () => {}
|
||||||
|
getPrevByNumber = () => ({
|
||||||
|
refSerialize: 'xxxxx',
|
||||||
|
ref: 'xxxx',
|
||||||
|
options: { callback: true, nested: ['1', '2'] },
|
||||||
|
})
|
||||||
|
saveLog = () => {}
|
||||||
|
}
|
||||||
|
|
||||||
|
test(`[CoreClass] Probando instanciamiento de clase`, async () => {
|
||||||
|
const setting = {
|
||||||
|
flow: new MockFlow(),
|
||||||
|
database: new MockDBA(),
|
||||||
|
provider: new MockProvider(),
|
||||||
|
}
|
||||||
|
const bot = await createBot(setting)
|
||||||
|
assert.is(bot instanceof CoreClass, true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test(`[CoreClass createFlow] Probando instanciamiento de clase`, async () => {
|
||||||
|
const mockCreateFlow = createFlow([])
|
||||||
|
assert.is(mockCreateFlow instanceof FlowClass, true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test(`[CoreClass createProvider] Probando instanciamiento de clase`, async () => {
|
||||||
|
const mockCreateProvider = createProvider(MockProvider)
|
||||||
|
assert.is(mockCreateProvider instanceof ProviderClass, true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test(`[Bot] Eventos 'require_action,ready,auth_failure,message '`, async () => {
|
||||||
|
let responseEvents = {}
|
||||||
|
|
||||||
|
const MOCK_EVENTS = {
|
||||||
|
require_action: {
|
||||||
|
instructions: 'Debes...',
|
||||||
|
},
|
||||||
|
ready: true,
|
||||||
|
auth_failure: {
|
||||||
|
instructions: 'Error...',
|
||||||
|
},
|
||||||
|
message: {
|
||||||
|
from: 'XXXXXX',
|
||||||
|
body: 'hola',
|
||||||
|
hasMedia: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const mockProvider = new MockProvider()
|
||||||
|
|
||||||
|
const setting = {
|
||||||
|
flow: new MockFlow(),
|
||||||
|
database: new MockDBA(),
|
||||||
|
provider: mockProvider,
|
||||||
|
}
|
||||||
|
await createBot(setting)
|
||||||
|
|
||||||
|
/// Escuchamos eventos
|
||||||
|
mockProvider.on('require_action', (r) => (responseEvents['require_action'] = r))
|
||||||
|
mockProvider.on('ready', (r) => (responseEvents['ready'] = r))
|
||||||
|
mockProvider.on('auth_failure', (r) => (responseEvents['auth_failure'] = r))
|
||||||
|
mockProvider.on('message', (r) => (responseEvents['message'] = r))
|
||||||
|
|
||||||
|
/// Emitimos eventos
|
||||||
|
mockProvider.delaySendMessage(0, 'require_action', MOCK_EVENTS.require_action)
|
||||||
|
mockProvider.delaySendMessage(0, 'ready', MOCK_EVENTS.ready)
|
||||||
|
mockProvider.delaySendMessage(0, 'auth_failure', MOCK_EVENTS.auth_failure)
|
||||||
|
mockProvider.delaySendMessage(0, 'message', MOCK_EVENTS.message)
|
||||||
|
|
||||||
|
await delay(0)
|
||||||
|
|
||||||
|
/// Testeamos eventos
|
||||||
|
assert.is(JSON.stringify(responseEvents.require_action), JSON.stringify(MOCK_EVENTS.require_action))
|
||||||
|
assert.is(responseEvents.ready, MOCK_EVENTS.ready)
|
||||||
|
|
||||||
|
assert.is(JSON.stringify(responseEvents.auth_failure), JSON.stringify(MOCK_EVENTS.auth_failure))
|
||||||
|
|
||||||
|
assert.is(JSON.stringify(responseEvents.message), JSON.stringify(MOCK_EVENTS.message))
|
||||||
|
})
|
||||||
|
|
||||||
|
test(`[Bot] Probando Flujos Internos`, async () => {
|
||||||
|
let responseEvents = {}
|
||||||
|
|
||||||
|
const MOCK_EVENTS = {
|
||||||
|
require_action: {
|
||||||
|
instructions: 'Debes...',
|
||||||
|
},
|
||||||
|
ready: true,
|
||||||
|
auth_failure: {
|
||||||
|
instructions: 'Error...',
|
||||||
|
},
|
||||||
|
message: {
|
||||||
|
from: 'XXXXXX',
|
||||||
|
body: 'hola',
|
||||||
|
hasMedia: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const mockProvider = new MockProvider()
|
||||||
|
|
||||||
|
const setting = {
|
||||||
|
flow: new MockFlow(),
|
||||||
|
database: new MockDBB(),
|
||||||
|
provider: mockProvider,
|
||||||
|
}
|
||||||
|
await createBot(setting)
|
||||||
|
|
||||||
|
/// Escuchamos eventos
|
||||||
|
mockProvider.on('require_action', (r) => (responseEvents['require_action'] = r))
|
||||||
|
mockProvider.on('ready', (r) => (responseEvents['ready'] = r))
|
||||||
|
mockProvider.on('auth_failure', (r) => (responseEvents['auth_failure'] = r))
|
||||||
|
mockProvider.on('message', (r) => (responseEvents['message'] = r))
|
||||||
|
|
||||||
|
/// Emitimos eventos
|
||||||
|
mockProvider.delaySendMessage(0, 'require_action', MOCK_EVENTS.require_action)
|
||||||
|
mockProvider.delaySendMessage(0, 'ready', MOCK_EVENTS.ready)
|
||||||
|
mockProvider.delaySendMessage(0, 'auth_failure', MOCK_EVENTS.auth_failure)
|
||||||
|
mockProvider.delaySendMessage(0, 'message', MOCK_EVENTS.message)
|
||||||
|
|
||||||
|
await delay(0)
|
||||||
|
|
||||||
|
/// Testeamos eventos
|
||||||
|
assert.is(JSON.stringify(responseEvents.require_action), JSON.stringify(MOCK_EVENTS.require_action))
|
||||||
|
assert.is(responseEvents.ready, MOCK_EVENTS.ready)
|
||||||
|
|
||||||
|
assert.is(JSON.stringify(responseEvents.auth_failure), JSON.stringify(MOCK_EVENTS.auth_failure))
|
||||||
|
|
||||||
|
assert.is(JSON.stringify(responseEvents.message), JSON.stringify(MOCK_EVENTS.message))
|
||||||
|
})
|
||||||
|
|
||||||
|
test(`[Bot] Probando Flujos Nested`, async () => {
|
||||||
|
let responseEvents = {}
|
||||||
|
|
||||||
|
const MOCK_EVENTS = {
|
||||||
|
require_action: {
|
||||||
|
instructions: 'Debes...',
|
||||||
|
},
|
||||||
|
ready: true,
|
||||||
|
auth_failure: {
|
||||||
|
instructions: 'Error...',
|
||||||
|
},
|
||||||
|
message: {
|
||||||
|
from: 'XXXXXX',
|
||||||
|
body: 'hola',
|
||||||
|
hasMedia: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const mockProvider = new MockProvider()
|
||||||
|
|
||||||
|
const setting = {
|
||||||
|
flow: new MockFlow(),
|
||||||
|
database: new MockDBC(),
|
||||||
|
provider: mockProvider,
|
||||||
|
}
|
||||||
|
const botInstance = await createBot(setting)
|
||||||
|
|
||||||
|
botInstance.sendProviderAndSave('xxxxx', 'xxxxx')
|
||||||
|
botInstance.continue('xxxxx', 'xxxxx')
|
||||||
|
/// Escuchamos eventos
|
||||||
|
mockProvider.on('require_action', (r) => (responseEvents['require_action'] = r))
|
||||||
|
mockProvider.on('ready', (r) => (responseEvents['ready'] = r))
|
||||||
|
mockProvider.on('auth_failure', (r) => (responseEvents['auth_failure'] = r))
|
||||||
|
mockProvider.on('message', (r) => (responseEvents['message'] = r))
|
||||||
|
|
||||||
|
/// Emitimos eventos
|
||||||
|
mockProvider.delaySendMessage(0, 'require_action', MOCK_EVENTS.require_action)
|
||||||
|
mockProvider.delaySendMessage(0, 'ready', MOCK_EVENTS.ready)
|
||||||
|
mockProvider.delaySendMessage(0, 'auth_failure', MOCK_EVENTS.auth_failure)
|
||||||
|
mockProvider.delaySendMessage(0, 'message', MOCK_EVENTS.message)
|
||||||
|
|
||||||
|
await delay(0)
|
||||||
|
|
||||||
|
/// Testeamos eventos
|
||||||
|
assert.is(JSON.stringify(responseEvents.require_action), JSON.stringify(MOCK_EVENTS.require_action))
|
||||||
|
assert.is(responseEvents.ready, MOCK_EVENTS.ready)
|
||||||
|
|
||||||
|
assert.is(JSON.stringify(responseEvents.auth_failure), JSON.stringify(MOCK_EVENTS.auth_failure))
|
||||||
|
|
||||||
|
assert.is(JSON.stringify(responseEvents.message), JSON.stringify(MOCK_EVENTS.message))
|
||||||
|
})
|
||||||
|
|
||||||
|
test.run()
|
||||||
|
|
||||||
|
function delay(ms) {
|
||||||
|
return new Promise((res) => setTimeout(res, ms))
|
||||||
|
}
|
||||||
28
packages/bot/tests/flow.class.test.js
Normal file
28
packages/bot/tests/flow.class.test.js
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
const { test } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const FlowClass = require('../io/flow.class')
|
||||||
|
const { addKeyword } = require('../index')
|
||||||
|
|
||||||
|
test(`[FlowClass] Probando instanciamiento de clase`, async () => {
|
||||||
|
const MOCK_FLOW = addKeyword('hola').addAnswer('Buenas!')
|
||||||
|
const flowClass = new FlowClass([MOCK_FLOW])
|
||||||
|
assert.is(flowClass instanceof FlowClass, true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test(`[FlowClass] Probando find`, async () => {
|
||||||
|
const MOCK_FLOW = addKeyword('hola').addAnswer('Buenas!')
|
||||||
|
const flowClass = new FlowClass([MOCK_FLOW])
|
||||||
|
|
||||||
|
flowClass.find('hola')
|
||||||
|
assert.is(flowClass instanceof FlowClass, true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test(`[FlowClass] Probando findBySerialize`, async () => {
|
||||||
|
const MOCK_FLOW = addKeyword('hola').addAnswer('Buenas!')
|
||||||
|
const flowClass = new FlowClass([MOCK_FLOW])
|
||||||
|
|
||||||
|
flowClass.findBySerialize('')
|
||||||
|
assert.is(flowClass instanceof FlowClass, true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test.run()
|
||||||
153
packages/bot/tests/methods.test.js
Normal file
153
packages/bot/tests/methods.test.js
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
const { test } = require('uvu')
|
||||||
|
const assert = require('uvu/assert')
|
||||||
|
const { generateRefSerialize } = require('../utils/hash')
|
||||||
|
const { addKeyword, addAnswer, toSerialize } = require('../io/methods')
|
||||||
|
|
||||||
|
test('Debere probar las propeidades', () => {
|
||||||
|
const ARRANGE = {
|
||||||
|
keyword: 'hola!',
|
||||||
|
}
|
||||||
|
const MAIN_CTX = addKeyword(ARRANGE.keyword)
|
||||||
|
|
||||||
|
assert.type(MAIN_CTX.addAnswer, 'function')
|
||||||
|
assert.is(MAIN_CTX.ctx.keyword, ARRANGE.keyword)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Debere probar las propeidades array', () => {
|
||||||
|
const ARRANGE = {
|
||||||
|
keyword: ['hola!', 'ole'],
|
||||||
|
}
|
||||||
|
const MAIN_CTX = addKeyword(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', () => {
|
||||||
|
const ARRANGE = {
|
||||||
|
keyword: ['hola!', 'ole'],
|
||||||
|
}
|
||||||
|
const MAIN_CTX = addKeyword(ARRANGE.keyword).addAnswer('Segundo!').addAnswer('Segundo!').toJson()
|
||||||
|
|
||||||
|
const [ANSWER_A] = MAIN_CTX
|
||||||
|
|
||||||
|
assert.is(
|
||||||
|
toSerialize(MAIN_CTX)[0].refSerialize,
|
||||||
|
generateRefSerialize({
|
||||||
|
index: 0,
|
||||||
|
answer: ANSWER_A.answer,
|
||||||
|
keyword: ANSWER_A.keyword,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Debere probar el paso de contexto', () => {
|
||||||
|
const ARRANGE = {
|
||||||
|
keyword: 'hola!',
|
||||||
|
answer: 'Bienvenido',
|
||||||
|
}
|
||||||
|
const CTX_A = addKeyword(ARRANGE.keyword)
|
||||||
|
const CTX_B = addAnswer(CTX_A)(ARRANGE.answer)
|
||||||
|
|
||||||
|
assert.is(CTX_A.ctx.keyword, ARRANGE.keyword)
|
||||||
|
assert.is(CTX_B.ctx.keyword, ARRANGE.keyword)
|
||||||
|
assert.is(CTX_B.ctx.answer, ARRANGE.answer)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Debere probar la anidación', () => {
|
||||||
|
const ARRANGE = {
|
||||||
|
keyword: 'hola!',
|
||||||
|
answer_A: 'Bienvenido',
|
||||||
|
answer_B: 'Continuar',
|
||||||
|
}
|
||||||
|
const MAIN_CTX = addKeyword(ARRANGE.keyword).addAnswer(ARRANGE.answer_A).addAnswer(ARRANGE.answer_B)
|
||||||
|
|
||||||
|
assert.is(MAIN_CTX.ctx.answer, ARRANGE.answer_B)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Debere probar las poptions', () => {
|
||||||
|
const MAIN_CTX = addKeyword('etc', { sensitive: false })
|
||||||
|
|
||||||
|
assert.is(MAIN_CTX.ctx.options.sensitive, false)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Debere probar las addAnswer', () => {
|
||||||
|
const MOCK_OPT = {
|
||||||
|
media: 'http://image.mock/mock.png',
|
||||||
|
buttons: [1],
|
||||||
|
}
|
||||||
|
const MAIN_CTX = addKeyword('hola').addAnswer('etc', MOCK_OPT)
|
||||||
|
|
||||||
|
assert.is(MAIN_CTX.ctx.options.media, MOCK_OPT.media)
|
||||||
|
assert.is(MAIN_CTX.ctx.options.buttons.length, 1)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Debere probar error las addAnswer', () => {
|
||||||
|
const MOCK_OPT = {
|
||||||
|
media: { a: 1, b: [] },
|
||||||
|
buttons: 'test',
|
||||||
|
}
|
||||||
|
const MAIN_CTX = addKeyword('hola').addAnswer('etc', MOCK_OPT)
|
||||||
|
|
||||||
|
assert.is(MAIN_CTX.ctx.options.media, null)
|
||||||
|
assert.is(MAIN_CTX.ctx.options.buttons.length, 0)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Obtener toJson', () => {
|
||||||
|
const [ctxA, ctxB, ctxC] = addKeyword('hola').addAnswer('pera!').addAnswer('chao').toJson()
|
||||||
|
|
||||||
|
assert.is(ctxA.keyword, 'hola')
|
||||||
|
assert.match(ctxA.ref, /^key_/)
|
||||||
|
|
||||||
|
assert.is(ctxB.answer, 'pera!')
|
||||||
|
assert.match(ctxB.ref, /^ans_/)
|
||||||
|
|
||||||
|
assert.is(ctxC.answer, 'chao')
|
||||||
|
assert.match(ctxC.ref, /^ans_/)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('addKeyword toJson con sensitive', () => {
|
||||||
|
const [ctxA] = addKeyword('hola').toJson()
|
||||||
|
assert.is(ctxA.options.sensitive, false)
|
||||||
|
const [ctxB] = addKeyword('hola', { sensitive: true }).toJson()
|
||||||
|
assert.is(ctxB.options.sensitive, true)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('addAnswer toJson con IMG', () => {
|
||||||
|
const [, ctxB, ctxC] = addKeyword('hola')
|
||||||
|
.addAnswer('bye!', {
|
||||||
|
media: 'http://mock.img/file-a.png',
|
||||||
|
})
|
||||||
|
.addAnswer('otro!', {
|
||||||
|
media: 'http://mock.img/file-b.png',
|
||||||
|
})
|
||||||
|
.toJson()
|
||||||
|
|
||||||
|
assert.is(ctxB.options.media, 'http://mock.img/file-a.png')
|
||||||
|
assert.is(ctxC.options.media, 'http://mock.img/file-b.png')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('addAnswer toJson con BUTTONS', () => {
|
||||||
|
const [, ctxB] = addKeyword('hola')
|
||||||
|
.addAnswer('mis opciones!', {
|
||||||
|
buttons: [{ body: 'BTN_1' }, { body: 'BTN_2' }],
|
||||||
|
})
|
||||||
|
.toJson()
|
||||||
|
|
||||||
|
assert.is(ctxB.options.buttons.length, 2)
|
||||||
|
|
||||||
|
const [btnA, btnB] = ctxB.options.buttons
|
||||||
|
|
||||||
|
assert.is(btnA.body, 'BTN_1')
|
||||||
|
assert.is(btnB.body, 'BTN_2')
|
||||||
|
})
|
||||||
|
|
||||||
|
test.run()
|
||||||
3
packages/bot/utils/delay.js
Normal file
3
packages/bot/utils/delay.js
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
const delay = (miliseconds) => new Promise((res) => setTimeout(res, miliseconds))
|
||||||
|
|
||||||
|
module.exports = { delay }
|
||||||
23
packages/bot/utils/flattener.js
Normal file
23
packages/bot/utils/flattener.js
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
const flatObject = (listArray = []) => {
|
||||||
|
const cbNestedList = Array.isArray(listArray) ? listArray : []
|
||||||
|
|
||||||
|
if (!listArray.length) return {}
|
||||||
|
|
||||||
|
const cbNestedObj = cbNestedList.map(({ ctx }) => ctx?.callbacks).filter((i) => !!i)
|
||||||
|
const queueCb = cbNestedObj.reduce((acc, current) => {
|
||||||
|
const getKeys = Object.keys(current)
|
||||||
|
const parse = getKeys.map((icb, i) => ({
|
||||||
|
[icb]: Object.values(current)[i],
|
||||||
|
}))
|
||||||
|
return [...acc, ...parse]
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const flatObj = {}
|
||||||
|
for (const iteration of queueCb) {
|
||||||
|
const [keyCb] = Object.keys(iteration)
|
||||||
|
flatObj[keyCb] = iteration[keyCb]
|
||||||
|
}
|
||||||
|
return flatObj
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { flatObject }
|
||||||
21
packages/bot/utils/hash.js
Normal file
21
packages/bot/utils/hash.js
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
const crypto = require('crypto')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generamos un UUID unico con posibilidad de tener un prefijo
|
||||||
|
* @param {*} prefix
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const generateRef = (prefix = false) => {
|
||||||
|
const id = crypto.randomUUID()
|
||||||
|
return prefix ? `${prefix}_${id}` : id
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Genera un HASH MD5
|
||||||
|
* @param {*} param0
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
const generateRefSerialize = ({ index, answer, keyword }) =>
|
||||||
|
crypto.createHash('md5').update(JSON.stringify({ index, answer, keyword })).digest('hex')
|
||||||
|
|
||||||
|
module.exports = { generateRef, generateRefSerialize }
|
||||||
12
packages/bot/utils/interactive.js
Normal file
12
packages/bot/utils/interactive.js
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
const { yellow, bgRed } = require('kleur')
|
||||||
|
const NODE_ENV = process.env.NODE_ENV || 'dev'
|
||||||
|
const printer = (message, title) => {
|
||||||
|
if (NODE_ENV !== 'test') {
|
||||||
|
// console.clear()
|
||||||
|
if (title) console.log(bgRed(`${title}`))
|
||||||
|
console.log(yellow(Array.isArray(message) ? message.join('\n') : message))
|
||||||
|
console.log(``)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { printer }
|
||||||
46
packages/bot/utils/queue.js
Normal file
46
packages/bot/utils/queue.js
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
class Queue {
|
||||||
|
queue = []
|
||||||
|
pendingPromise = false
|
||||||
|
|
||||||
|
enqueue(promise) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
this.queue.push({
|
||||||
|
promise,
|
||||||
|
resolve,
|
||||||
|
reject,
|
||||||
|
})
|
||||||
|
this.dequeue()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
dequeue() {
|
||||||
|
if (this.workingOnPromise) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
const item = this.queue.shift()
|
||||||
|
if (!item) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
this.workingOnPromise = true
|
||||||
|
item.promise()
|
||||||
|
.then((value) => {
|
||||||
|
this.workingOnPromise = false
|
||||||
|
item.resolve(value)
|
||||||
|
this.dequeue()
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
this.workingOnPromise = false
|
||||||
|
item.reject(err)
|
||||||
|
this.dequeue()
|
||||||
|
})
|
||||||
|
} catch (err) {
|
||||||
|
this.workingOnPromise = false
|
||||||
|
item.reject(err)
|
||||||
|
this.dequeue()
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = Queue
|
||||||
20
packages/cli/README.md
Normal file
20
packages/cli/README.md
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
# @bot-whatsapp/cli
|
||||||
|
|
||||||
|
- [x] Revisar version de NODE
|
||||||
|
- [x] Revisar OS
|
||||||
|
- [x] Obtener Package Manager
|
||||||
|
- [x] Revisar las libreria de WhatsappWeb para obtener version reciente
|
||||||
|
- [x] Opcion interactiva de limpiar session
|
||||||
|
- [x] Opcion de generar `json` con la configuracion
|
||||||
|
- [x] Agregar `rollup` para limpiar el codigo
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Comunidad**
|
||||||
|
|
||||||
|
> Forma parte de este proyecto.
|
||||||
|
|
||||||
|
- [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)
|
||||||
3
packages/cli/bin/cli.js
Executable file
3
packages/cli/bin/cli.js
Executable file
@@ -0,0 +1,3 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
const index = require('../lib/cli/bundle.cli.cjs')
|
||||||
|
index.startInteractive()
|
||||||
59
packages/cli/check/index.js
Normal file
59
packages/cli/check/index.js
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
const { red, yellow, green, bgCyan } = require('kleur')
|
||||||
|
const { exec } = require('node:child_process')
|
||||||
|
|
||||||
|
const checkNodeVersion = () => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
console.log(bgCyan('🚀 Revisando tu Node.js'))
|
||||||
|
const version = process.version
|
||||||
|
const majorVersion = parseInt(version.replace('v', '').split('.').shift())
|
||||||
|
if (majorVersion < 16) {
|
||||||
|
console.error(red(`🔴 Se require Node.js 16 o superior. Actualmente esta ejecutando Node.js ${version}`))
|
||||||
|
console.log(``)
|
||||||
|
reject('ERROR_NODE')
|
||||||
|
}
|
||||||
|
console.log(green(`Node.js: ${version} compatible ✅`))
|
||||||
|
console.log(``)
|
||||||
|
resolve()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const checkOs = () => {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
console.log(bgCyan('🙂 Revisando tu sistema operativo'))
|
||||||
|
const os = process.platform
|
||||||
|
if (!os.includes('win32')) {
|
||||||
|
const messages = [
|
||||||
|
`El sistema operativo actual (${os}) posiblemente requiera`,
|
||||||
|
`una configuración adicional referente al puppeteer`,
|
||||||
|
``,
|
||||||
|
`Recuerda pasar por el WIKI`,
|
||||||
|
`🔗 https://github.com/leifermendez/bot-whatsapp/wiki/Instalación`,
|
||||||
|
``,
|
||||||
|
]
|
||||||
|
|
||||||
|
console.log(yellow(messages.join(' \n')))
|
||||||
|
}
|
||||||
|
console.log(green(`OS: compatible ✅`))
|
||||||
|
console.log(``)
|
||||||
|
resolve()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const checkGit = () => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
console.log(bgCyan('🤓 Revisando GIT'))
|
||||||
|
exec('git --version', (error) => {
|
||||||
|
if (error) {
|
||||||
|
console.error(red(`🔴 Se require instalar GIT`))
|
||||||
|
console.log(``)
|
||||||
|
reject('ERROR_GIT')
|
||||||
|
} else {
|
||||||
|
console.log(green(`Git: Compatible ✅`))
|
||||||
|
console.log(``)
|
||||||
|
resolve()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { checkNodeVersion, checkOs, checkGit }
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user