machinaos 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (288) hide show
  1. package/.env.template +71 -0
  2. package/LICENSE +21 -0
  3. package/README.md +87 -0
  4. package/bin/cli.js +159 -0
  5. package/client/.dockerignore +45 -0
  6. package/client/Dockerfile +68 -0
  7. package/client/eslint.config.js +29 -0
  8. package/client/index.html +13 -0
  9. package/client/nginx.conf +66 -0
  10. package/client/package.json +48 -0
  11. package/client/src/App.tsx +27 -0
  12. package/client/src/Dashboard.tsx +1173 -0
  13. package/client/src/ParameterPanel.tsx +301 -0
  14. package/client/src/components/AIAgentNode.tsx +321 -0
  15. package/client/src/components/APIKeyValidator.tsx +118 -0
  16. package/client/src/components/ClaudeChatModelNode.tsx +18 -0
  17. package/client/src/components/ConditionalEdge.tsx +189 -0
  18. package/client/src/components/CredentialsModal.tsx +306 -0
  19. package/client/src/components/EdgeConditionEditor.tsx +443 -0
  20. package/client/src/components/GeminiChatModelNode.tsx +18 -0
  21. package/client/src/components/GenericNode.tsx +357 -0
  22. package/client/src/components/LocationParameterPanel.tsx +154 -0
  23. package/client/src/components/ModelNode.tsx +286 -0
  24. package/client/src/components/OpenAIChatModelNode.tsx +18 -0
  25. package/client/src/components/OutputPanel.tsx +471 -0
  26. package/client/src/components/ParameterRenderer.tsx +1874 -0
  27. package/client/src/components/SkillEditorModal.tsx +417 -0
  28. package/client/src/components/SquareNode.tsx +797 -0
  29. package/client/src/components/StartNode.tsx +250 -0
  30. package/client/src/components/ToolkitNode.tsx +365 -0
  31. package/client/src/components/TriggerNode.tsx +463 -0
  32. package/client/src/components/auth/LoginPage.tsx +247 -0
  33. package/client/src/components/auth/ProtectedRoute.tsx +59 -0
  34. package/client/src/components/base/BaseChatModelNode.tsx +271 -0
  35. package/client/src/components/icons/AIProviderIcons.tsx +50 -0
  36. package/client/src/components/maps/GoogleMapsPicker.tsx +137 -0
  37. package/client/src/components/maps/MapsPreviewPanel.tsx +110 -0
  38. package/client/src/components/maps/index.ts +26 -0
  39. package/client/src/components/parameterPanel/InputSection.tsx +1094 -0
  40. package/client/src/components/parameterPanel/LocationPanelLayout.tsx +65 -0
  41. package/client/src/components/parameterPanel/MapsSection.tsx +92 -0
  42. package/client/src/components/parameterPanel/MiddleSection.tsx +571 -0
  43. package/client/src/components/parameterPanel/OutputSection.tsx +81 -0
  44. package/client/src/components/parameterPanel/ParameterPanelLayout.tsx +82 -0
  45. package/client/src/components/parameterPanel/ToolSchemaEditor.tsx +436 -0
  46. package/client/src/components/parameterPanel/index.ts +42 -0
  47. package/client/src/components/shared/DataPanel.tsx +142 -0
  48. package/client/src/components/shared/JSONTreeRenderer.tsx +106 -0
  49. package/client/src/components/ui/AIResultModal.tsx +204 -0
  50. package/client/src/components/ui/AndroidSettingsPanel.tsx +401 -0
  51. package/client/src/components/ui/CodeEditor.tsx +81 -0
  52. package/client/src/components/ui/CollapsibleSection.tsx +88 -0
  53. package/client/src/components/ui/ComponentItem.tsx +154 -0
  54. package/client/src/components/ui/ComponentPalette.tsx +321 -0
  55. package/client/src/components/ui/ConsolePanel.tsx +1074 -0
  56. package/client/src/components/ui/ErrorBoundary.tsx +196 -0
  57. package/client/src/components/ui/InputNodesPanel.tsx +204 -0
  58. package/client/src/components/ui/MapSelector.tsx +314 -0
  59. package/client/src/components/ui/Modal.tsx +149 -0
  60. package/client/src/components/ui/NodeContextMenu.tsx +192 -0
  61. package/client/src/components/ui/NodeOutputPanel.tsx +1150 -0
  62. package/client/src/components/ui/OutputDisplayPanel.tsx +381 -0
  63. package/client/src/components/ui/SettingsPanel.tsx +243 -0
  64. package/client/src/components/ui/TopToolbar.tsx +736 -0
  65. package/client/src/components/ui/WhatsAppSettingsPanel.tsx +345 -0
  66. package/client/src/components/ui/WorkflowSidebar.tsx +294 -0
  67. package/client/src/config/antdTheme.ts +186 -0
  68. package/client/src/config/api.ts +54 -0
  69. package/client/src/contexts/AuthContext.tsx +221 -0
  70. package/client/src/contexts/ThemeContext.tsx +42 -0
  71. package/client/src/contexts/WebSocketContext.tsx +1971 -0
  72. package/client/src/factories/baseChatModelFactory.ts +256 -0
  73. package/client/src/hooks/useAndroidOperations.ts +164 -0
  74. package/client/src/hooks/useApiKeyValidation.ts +107 -0
  75. package/client/src/hooks/useApiKeys.ts +238 -0
  76. package/client/src/hooks/useAppTheme.ts +17 -0
  77. package/client/src/hooks/useComponentPalette.ts +51 -0
  78. package/client/src/hooks/useCopyPaste.ts +155 -0
  79. package/client/src/hooks/useDragAndDrop.ts +124 -0
  80. package/client/src/hooks/useDragVariable.ts +88 -0
  81. package/client/src/hooks/useExecution.ts +313 -0
  82. package/client/src/hooks/useParameterPanel.ts +176 -0
  83. package/client/src/hooks/useReactFlowNodes.ts +189 -0
  84. package/client/src/hooks/useToolSchema.ts +209 -0
  85. package/client/src/hooks/useWhatsApp.ts +196 -0
  86. package/client/src/hooks/useWorkflowManagement.ts +46 -0
  87. package/client/src/index.css +315 -0
  88. package/client/src/main.tsx +19 -0
  89. package/client/src/nodeDefinitions/aiAgentNodes.ts +336 -0
  90. package/client/src/nodeDefinitions/aiModelNodes.ts +340 -0
  91. package/client/src/nodeDefinitions/androidDeviceNodes.ts +140 -0
  92. package/client/src/nodeDefinitions/androidServiceNodes.ts +383 -0
  93. package/client/src/nodeDefinitions/chatNodes.ts +135 -0
  94. package/client/src/nodeDefinitions/codeNodes.ts +54 -0
  95. package/client/src/nodeDefinitions/documentNodes.ts +379 -0
  96. package/client/src/nodeDefinitions/index.ts +15 -0
  97. package/client/src/nodeDefinitions/locationNodes.ts +463 -0
  98. package/client/src/nodeDefinitions/schedulerNodes.ts +220 -0
  99. package/client/src/nodeDefinitions/skillNodes.ts +211 -0
  100. package/client/src/nodeDefinitions/toolNodes.ts +198 -0
  101. package/client/src/nodeDefinitions/utilityNodes.ts +284 -0
  102. package/client/src/nodeDefinitions/whatsappNodes.ts +865 -0
  103. package/client/src/nodeDefinitions/workflowNodes.ts +41 -0
  104. package/client/src/nodeDefinitions.ts +104 -0
  105. package/client/src/schemas/workflowSchema.ts +264 -0
  106. package/client/src/services/dynamicParameterService.ts +96 -0
  107. package/client/src/services/execution/aiAgentExecutionService.ts +35 -0
  108. package/client/src/services/executionService.ts +232 -0
  109. package/client/src/services/workflowApi.ts +91 -0
  110. package/client/src/store/useAppStore.ts +582 -0
  111. package/client/src/styles/theme.ts +508 -0
  112. package/client/src/styles/zIndex.ts +17 -0
  113. package/client/src/types/ComponentTypes.ts +39 -0
  114. package/client/src/types/EdgeCondition.ts +231 -0
  115. package/client/src/types/INodeProperties.ts +288 -0
  116. package/client/src/types/NodeTypes.ts +28 -0
  117. package/client/src/utils/formatters.ts +33 -0
  118. package/client/src/utils/googleMapsLoader.ts +140 -0
  119. package/client/src/utils/locationUtils.ts +85 -0
  120. package/client/src/utils/nodeUtils.ts +31 -0
  121. package/client/src/utils/workflow.ts +30 -0
  122. package/client/src/utils/workflowExport.ts +120 -0
  123. package/client/src/vite-env.d.ts +12 -0
  124. package/client/tailwind.config.js +60 -0
  125. package/client/tsconfig.json +25 -0
  126. package/client/tsconfig.node.json +11 -0
  127. package/client/vite.config.js +35 -0
  128. package/docker-compose.prod.yml +107 -0
  129. package/docker-compose.yml +104 -0
  130. package/docs-MachinaOs/README.md +85 -0
  131. package/docs-MachinaOs/deployment/docker.mdx +228 -0
  132. package/docs-MachinaOs/deployment/production.mdx +345 -0
  133. package/docs-MachinaOs/docs.json +75 -0
  134. package/docs-MachinaOs/faq.mdx +309 -0
  135. package/docs-MachinaOs/favicon.svg +5 -0
  136. package/docs-MachinaOs/installation.mdx +160 -0
  137. package/docs-MachinaOs/introduction.mdx +114 -0
  138. package/docs-MachinaOs/logo/dark.svg +6 -0
  139. package/docs-MachinaOs/logo/light.svg +6 -0
  140. package/docs-MachinaOs/nodes/ai-agent.mdx +216 -0
  141. package/docs-MachinaOs/nodes/ai-models.mdx +240 -0
  142. package/docs-MachinaOs/nodes/android.mdx +411 -0
  143. package/docs-MachinaOs/nodes/overview.mdx +181 -0
  144. package/docs-MachinaOs/nodes/schedulers.mdx +316 -0
  145. package/docs-MachinaOs/nodes/webhooks.mdx +330 -0
  146. package/docs-MachinaOs/nodes/whatsapp.mdx +305 -0
  147. package/docs-MachinaOs/quickstart.mdx +119 -0
  148. package/docs-MachinaOs/tutorials/ai-agent-workflow.mdx +177 -0
  149. package/docs-MachinaOs/tutorials/android-automation.mdx +242 -0
  150. package/docs-MachinaOs/tutorials/first-workflow.mdx +134 -0
  151. package/docs-MachinaOs/tutorials/whatsapp-automation.mdx +185 -0
  152. package/nul +0 -0
  153. package/package.json +70 -0
  154. package/scripts/build.js +158 -0
  155. package/scripts/check-ports.ps1 +33 -0
  156. package/scripts/clean.js +40 -0
  157. package/scripts/docker.js +93 -0
  158. package/scripts/kill-port.ps1 +154 -0
  159. package/scripts/start.js +210 -0
  160. package/scripts/stop.js +325 -0
  161. package/server/.dockerignore +44 -0
  162. package/server/Dockerfile +45 -0
  163. package/server/constants.py +249 -0
  164. package/server/core/__init__.py +1 -0
  165. package/server/core/cache.py +461 -0
  166. package/server/core/config.py +128 -0
  167. package/server/core/container.py +99 -0
  168. package/server/core/database.py +1211 -0
  169. package/server/core/logging.py +314 -0
  170. package/server/main.py +289 -0
  171. package/server/middleware/__init__.py +5 -0
  172. package/server/middleware/auth.py +89 -0
  173. package/server/models/__init__.py +1 -0
  174. package/server/models/auth.py +52 -0
  175. package/server/models/cache.py +24 -0
  176. package/server/models/database.py +211 -0
  177. package/server/models/nodes.py +455 -0
  178. package/server/package.json +9 -0
  179. package/server/pyproject.toml +72 -0
  180. package/server/requirements.txt +83 -0
  181. package/server/routers/__init__.py +1 -0
  182. package/server/routers/android.py +294 -0
  183. package/server/routers/auth.py +203 -0
  184. package/server/routers/database.py +151 -0
  185. package/server/routers/maps.py +142 -0
  186. package/server/routers/nodejs_compat.py +289 -0
  187. package/server/routers/webhook.py +90 -0
  188. package/server/routers/websocket.py +2127 -0
  189. package/server/routers/whatsapp.py +761 -0
  190. package/server/routers/workflow.py +200 -0
  191. package/server/services/__init__.py +1 -0
  192. package/server/services/ai.py +2415 -0
  193. package/server/services/android/__init__.py +27 -0
  194. package/server/services/android/broadcaster.py +114 -0
  195. package/server/services/android/client.py +608 -0
  196. package/server/services/android/manager.py +78 -0
  197. package/server/services/android/protocol.py +165 -0
  198. package/server/services/android_service.py +588 -0
  199. package/server/services/auth.py +131 -0
  200. package/server/services/chat_client.py +160 -0
  201. package/server/services/deployment/__init__.py +12 -0
  202. package/server/services/deployment/manager.py +706 -0
  203. package/server/services/deployment/state.py +47 -0
  204. package/server/services/deployment/triggers.py +275 -0
  205. package/server/services/event_waiter.py +785 -0
  206. package/server/services/execution/__init__.py +77 -0
  207. package/server/services/execution/cache.py +769 -0
  208. package/server/services/execution/conditions.py +373 -0
  209. package/server/services/execution/dlq.py +132 -0
  210. package/server/services/execution/executor.py +1351 -0
  211. package/server/services/execution/models.py +531 -0
  212. package/server/services/execution/recovery.py +235 -0
  213. package/server/services/handlers/__init__.py +126 -0
  214. package/server/services/handlers/ai.py +355 -0
  215. package/server/services/handlers/android.py +260 -0
  216. package/server/services/handlers/code.py +278 -0
  217. package/server/services/handlers/document.py +598 -0
  218. package/server/services/handlers/http.py +193 -0
  219. package/server/services/handlers/polyglot.py +105 -0
  220. package/server/services/handlers/tools.py +845 -0
  221. package/server/services/handlers/triggers.py +107 -0
  222. package/server/services/handlers/utility.py +822 -0
  223. package/server/services/handlers/whatsapp.py +476 -0
  224. package/server/services/maps.py +289 -0
  225. package/server/services/memory_store.py +103 -0
  226. package/server/services/node_executor.py +375 -0
  227. package/server/services/parameter_resolver.py +218 -0
  228. package/server/services/polyglot_client.py +169 -0
  229. package/server/services/scheduler.py +155 -0
  230. package/server/services/skill_loader.py +417 -0
  231. package/server/services/status_broadcaster.py +826 -0
  232. package/server/services/temporal/__init__.py +23 -0
  233. package/server/services/temporal/activities.py +344 -0
  234. package/server/services/temporal/client.py +76 -0
  235. package/server/services/temporal/executor.py +147 -0
  236. package/server/services/temporal/worker.py +251 -0
  237. package/server/services/temporal/workflow.py +355 -0
  238. package/server/services/temporal/ws_client.py +236 -0
  239. package/server/services/text.py +111 -0
  240. package/server/services/user_auth.py +172 -0
  241. package/server/services/websocket_client.py +29 -0
  242. package/server/services/workflow.py +597 -0
  243. package/server/skills/android-skill/SKILL.md +82 -0
  244. package/server/skills/assistant-personality/SKILL.md +45 -0
  245. package/server/skills/code-skill/SKILL.md +140 -0
  246. package/server/skills/http-skill/SKILL.md +161 -0
  247. package/server/skills/maps-skill/SKILL.md +170 -0
  248. package/server/skills/memory-skill/SKILL.md +154 -0
  249. package/server/skills/scheduler-skill/SKILL.md +84 -0
  250. package/server/skills/whatsapp-skill/SKILL.md +283 -0
  251. package/server/uv.lock +2916 -0
  252. package/server/whatsapp-rpc/.dockerignore +30 -0
  253. package/server/whatsapp-rpc/Dockerfile +44 -0
  254. package/server/whatsapp-rpc/Dockerfile.web +17 -0
  255. package/server/whatsapp-rpc/README.md +139 -0
  256. package/server/whatsapp-rpc/cli.js +95 -0
  257. package/server/whatsapp-rpc/configs/config.yaml +7 -0
  258. package/server/whatsapp-rpc/docker-compose.yml +35 -0
  259. package/server/whatsapp-rpc/docs/API.md +410 -0
  260. package/server/whatsapp-rpc/go.mod +67 -0
  261. package/server/whatsapp-rpc/go.sum +203 -0
  262. package/server/whatsapp-rpc/package.json +30 -0
  263. package/server/whatsapp-rpc/schema.json +1294 -0
  264. package/server/whatsapp-rpc/scripts/clean.cjs +66 -0
  265. package/server/whatsapp-rpc/scripts/cli.js +162 -0
  266. package/server/whatsapp-rpc/src/go/cmd/server/main.go +91 -0
  267. package/server/whatsapp-rpc/src/go/config/config.go +49 -0
  268. package/server/whatsapp-rpc/src/go/rpc/rpc.go +446 -0
  269. package/server/whatsapp-rpc/src/go/rpc/server.go +112 -0
  270. package/server/whatsapp-rpc/src/go/whatsapp/history.go +166 -0
  271. package/server/whatsapp-rpc/src/go/whatsapp/messages.go +390 -0
  272. package/server/whatsapp-rpc/src/go/whatsapp/service.go +2130 -0
  273. package/server/whatsapp-rpc/src/go/whatsapp/types.go +261 -0
  274. package/server/whatsapp-rpc/src/python/pyproject.toml +15 -0
  275. package/server/whatsapp-rpc/src/python/whatsapp_rpc/__init__.py +4 -0
  276. package/server/whatsapp-rpc/src/python/whatsapp_rpc/client.py +427 -0
  277. package/server/whatsapp-rpc/web/app.py +609 -0
  278. package/server/whatsapp-rpc/web/requirements.txt +6 -0
  279. package/server/whatsapp-rpc/web/rpc_client.py +427 -0
  280. package/server/whatsapp-rpc/web/static/openapi.yaml +59 -0
  281. package/server/whatsapp-rpc/web/templates/base.html +150 -0
  282. package/server/whatsapp-rpc/web/templates/contacts.html +240 -0
  283. package/server/whatsapp-rpc/web/templates/dashboard.html +320 -0
  284. package/server/whatsapp-rpc/web/templates/groups.html +328 -0
  285. package/server/whatsapp-rpc/web/templates/messages.html +465 -0
  286. package/server/whatsapp-rpc/web/templates/messaging.html +681 -0
  287. package/server/whatsapp-rpc/web/templates/send.html +259 -0
  288. package/server/whatsapp-rpc/web/templates/settings.html +459 -0
@@ -0,0 +1,240 @@
1
+ {% extends "base.html" %}
2
+ {% block title %}Contacts - WhatsApp Controller{% endblock %}
3
+
4
+ {% block content %}
5
+ <div class="space-y-8">
6
+ <!-- Page Header -->
7
+ <div>
8
+ <h1 class="text-3xl font-bold text-gray-900">Contacts</h1>
9
+ <p class="text-gray-600 mt-1">Check if phone numbers are on WhatsApp and view profile pictures</p>
10
+ </div>
11
+
12
+ <!-- Check Numbers Section -->
13
+ <div class="bg-white rounded-lg shadow-md p-6">
14
+ <h2 class="text-xl font-semibold mb-4">Check Phone Numbers</h2>
15
+ <p class="text-gray-600 mb-4">Enter phone numbers (one per line) to check if they are registered on WhatsApp</p>
16
+
17
+ <div class="space-y-4">
18
+ <div>
19
+ <label class="block text-sm font-medium text-gray-700 mb-2">Phone Numbers</label>
20
+ <textarea id="phone-numbers" rows="5"
21
+ placeholder="Enter phone numbers, one per line (without + prefix)&#10;e.g.&#10;919876543210&#10;14155551234"
22
+ class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-green-500"></textarea>
23
+ </div>
24
+
25
+ <button onclick="checkContacts()" class="px-6 py-2 bg-green-600 text-white rounded-lg hover:bg-green-700 flex items-center">
26
+ <svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
27
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"></path>
28
+ </svg>
29
+ Check Numbers
30
+ </button>
31
+ </div>
32
+
33
+ <!-- Results -->
34
+ <div id="check-results" class="mt-6 hidden">
35
+ <h3 class="font-semibold text-gray-800 mb-3">Results</h3>
36
+ <div id="results-container" class="space-y-2"></div>
37
+ </div>
38
+ </div>
39
+
40
+ <!-- Profile Picture Section -->
41
+ <div class="bg-white rounded-lg shadow-md p-6">
42
+ <h2 class="text-xl font-semibold mb-4">Profile Picture</h2>
43
+ <p class="text-gray-600 mb-4">View profile picture for a user or group</p>
44
+
45
+ <div class="flex flex-wrap gap-4 items-end">
46
+ <div class="flex-1 min-w-[250px]">
47
+ <label class="block text-sm font-medium text-gray-700 mb-2">JID (Phone or Group)</label>
48
+ <input type="text" id="profile-jid"
49
+ placeholder="919876543210@s.whatsapp.net or group@g.us"
50
+ class="w-full px-4 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-green-500">
51
+ </div>
52
+ <button onclick="getProfilePicture()" class="px-6 py-2 bg-blue-600 text-white rounded-lg hover:bg-blue-700 flex items-center">
53
+ <svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
54
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z"></path>
55
+ </svg>
56
+ Get Picture
57
+ </button>
58
+ </div>
59
+
60
+ <!-- Profile Picture Result -->
61
+ <div id="profile-result" class="mt-6 hidden">
62
+ <div id="profile-container" class="flex items-center space-x-4">
63
+ <img id="profile-image" src="" alt="Profile Picture" class="w-24 h-24 rounded-full object-cover border-4 border-gray-200">
64
+ <div>
65
+ <p id="profile-jid-display" class="font-semibold text-gray-800"></p>
66
+ <p id="profile-status" class="text-sm text-gray-600"></p>
67
+ </div>
68
+ </div>
69
+ <div id="profile-no-picture" class="hidden text-gray-500">
70
+ <svg class="w-24 h-24 text-gray-300 mx-auto" fill="currentColor" viewBox="0 0 24 24">
71
+ <path d="M24 20.993V24H0v-2.996A14.977 14.977 0 0112.004 15c4.904 0 9.26 2.354 11.996 5.993zM16.002 8.999a4 4 0 11-8 0 4 4 0 018 0z"></path>
72
+ </svg>
73
+ <p class="text-center mt-2">No profile picture available</p>
74
+ </div>
75
+ </div>
76
+ </div>
77
+
78
+ <!-- Presence Section -->
79
+ <div class="bg-white rounded-lg shadow-md p-6">
80
+ <h2 class="text-xl font-semibold mb-4">Presence Status</h2>
81
+ <p class="text-gray-600 mb-4">Set your online/offline presence status</p>
82
+
83
+ <div class="flex flex-wrap gap-4">
84
+ <button onclick="setPresence('available')" class="px-6 py-2 bg-green-600 text-white rounded-lg hover:bg-green-700 flex items-center">
85
+ <span class="w-3 h-3 bg-green-300 rounded-full mr-2"></span>
86
+ Set Online
87
+ </button>
88
+ <button onclick="setPresence('unavailable')" class="px-6 py-2 bg-gray-600 text-white rounded-lg hover:bg-gray-700 flex items-center">
89
+ <span class="w-3 h-3 bg-gray-400 rounded-full mr-2"></span>
90
+ Set Offline
91
+ </button>
92
+ </div>
93
+ <p id="presence-status" class="mt-3 text-sm text-gray-600"></p>
94
+ </div>
95
+ </div>
96
+ {% endblock %}
97
+
98
+ {% block scripts %}
99
+ <script>
100
+ // Listen for contact check result
101
+ socket.on('contact_check_result', function(data) {
102
+ const resultsSection = document.getElementById('check-results');
103
+ const container = document.getElementById('results-container');
104
+
105
+ resultsSection.classList.remove('hidden');
106
+
107
+ if (!data.success) {
108
+ container.innerHTML = `<div class="text-red-600">Error: ${data.error}</div>`;
109
+ return;
110
+ }
111
+
112
+ const results = data.data;
113
+ container.innerHTML = results.map(r => {
114
+ const statusClass = r.is_registered ? 'bg-green-100 border-green-300' : 'bg-red-100 border-red-300';
115
+ const statusIcon = r.is_registered
116
+ ? '<svg class="w-5 h-5 text-green-600" fill="currentColor" viewBox="0 0 20 20"><path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clip-rule="evenodd"></path></svg>'
117
+ : '<svg class="w-5 h-5 text-red-600" fill="currentColor" viewBox="0 0 20 20"><path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z" clip-rule="evenodd"></path></svg>';
118
+
119
+ let businessBadge = '';
120
+ if (r.is_business) {
121
+ businessBadge = `<span class="ml-2 px-2 py-1 bg-blue-100 text-blue-800 text-xs rounded">Business${r.business_name ? ': ' + escapeHtml(r.business_name) : ''}</span>`;
122
+ }
123
+
124
+ return `
125
+ <div class="flex items-center justify-between p-3 border rounded-lg ${statusClass}">
126
+ <div class="flex items-center space-x-3">
127
+ ${statusIcon}
128
+ <div>
129
+ <p class="font-medium">${escapeHtml(r.query)}</p>
130
+ ${r.is_registered ? `<p class="text-xs text-gray-600">${escapeHtml(r.jid || '')}</p>` : ''}
131
+ </div>
132
+ ${businessBadge}
133
+ </div>
134
+ <span class="text-sm font-medium ${r.is_registered ? 'text-green-700' : 'text-red-700'}">
135
+ ${r.is_registered ? 'On WhatsApp' : 'Not on WhatsApp'}
136
+ </span>
137
+ </div>
138
+ `;
139
+ }).join('');
140
+ });
141
+
142
+ // Listen for profile picture result
143
+ socket.on('contact_profile_pic_result', function(data) {
144
+ const resultSection = document.getElementById('profile-result');
145
+ const container = document.getElementById('profile-container');
146
+ const noPicture = document.getElementById('profile-no-picture');
147
+ const image = document.getElementById('profile-image');
148
+ const jidDisplay = document.getElementById('profile-jid-display');
149
+ const statusDisplay = document.getElementById('profile-status');
150
+
151
+ resultSection.classList.remove('hidden');
152
+
153
+ if (!data.success) {
154
+ container.classList.add('hidden');
155
+ noPicture.classList.remove('hidden');
156
+ noPicture.innerHTML = `<p class="text-red-600">Error: ${data.error}</p>`;
157
+ return;
158
+ }
159
+
160
+ jidDisplay.textContent = data.jid || '';
161
+
162
+ if (data.data.exists) {
163
+ container.classList.remove('hidden');
164
+ noPicture.classList.add('hidden');
165
+
166
+ if (data.data.data) {
167
+ // Base64 data available
168
+ image.src = 'data:image/jpeg;base64,' + data.data.data;
169
+ } else if (data.data.url) {
170
+ // URL available
171
+ image.src = data.data.url;
172
+ }
173
+ statusDisplay.textContent = 'Profile picture found';
174
+ } else {
175
+ container.classList.add('hidden');
176
+ noPicture.classList.remove('hidden');
177
+ noPicture.innerHTML = `
178
+ <svg class="w-24 h-24 text-gray-300 mx-auto" fill="currentColor" viewBox="0 0 24 24">
179
+ <path d="M24 20.993V24H0v-2.996A14.977 14.977 0 0112.004 15c4.904 0 9.26 2.354 11.996 5.993zM16.002 8.999a4 4 0 11-8 0 4 4 0 018 0z"></path>
180
+ </svg>
181
+ <p class="text-center mt-2">No profile picture available</p>
182
+ `;
183
+ }
184
+ });
185
+
186
+ // Listen for presence result
187
+ socket.on('presence_result', function(data) {
188
+ const status = document.getElementById('presence-status');
189
+ if (data.success) {
190
+ status.textContent = 'Presence updated successfully';
191
+ status.className = 'mt-3 text-sm text-green-600';
192
+ } else {
193
+ status.textContent = 'Failed: ' + data.error;
194
+ status.className = 'mt-3 text-sm text-red-600';
195
+ }
196
+ });
197
+
198
+ function checkContacts() {
199
+ const textarea = document.getElementById('phone-numbers');
200
+ const lines = textarea.value.split('\n').map(l => l.trim()).filter(l => l.length > 0);
201
+
202
+ if (lines.length === 0) {
203
+ alert('Please enter at least one phone number');
204
+ return;
205
+ }
206
+
207
+ socket.emit('contact_check', { phones: lines });
208
+ }
209
+
210
+ function getProfilePicture() {
211
+ const jid = document.getElementById('profile-jid').value.trim();
212
+
213
+ if (!jid) {
214
+ alert('Please enter a JID');
215
+ return;
216
+ }
217
+
218
+ // Auto-append @s.whatsapp.net if no @ present
219
+ let fullJid = jid;
220
+ if (!jid.includes('@')) {
221
+ fullJid = jid + '@s.whatsapp.net';
222
+ }
223
+
224
+ socket.emit('contact_profile_pic', { jid: fullJid, preview: false });
225
+ }
226
+
227
+ function setPresence(status) {
228
+ socket.emit('presence', { status: status });
229
+ document.getElementById('presence-status').textContent = 'Setting presence...';
230
+ document.getElementById('presence-status').className = 'mt-3 text-sm text-blue-600';
231
+ }
232
+
233
+ function escapeHtml(text) {
234
+ if (!text) return '';
235
+ const div = document.createElement('div');
236
+ div.textContent = text;
237
+ return div.innerHTML;
238
+ }
239
+ </script>
240
+ {% endblock %}
@@ -0,0 +1,320 @@
1
+ {% extends "base.html" %}
2
+
3
+ {% block title %}Dashboard - WhatsApp Controller{% endblock %}
4
+
5
+ {% block content %}
6
+ <div class="grid grid-cols-1 lg:grid-cols-3 gap-8">
7
+
8
+ <!-- Status Card -->
9
+ <div class="bg-white rounded-lg shadow-md p-6">
10
+ <h2 class="text-xl font-semibold mb-4 flex items-center">
11
+ <svg class="w-6 h-6 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
12
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"></path>
13
+ </svg>
14
+ Connection Status
15
+ </h2>
16
+
17
+ <div id="status-details" class="space-y-3">
18
+ <div class="flex justify-between items-center">
19
+ <span>Status:</span>
20
+ <span id="connection-status" class="font-medium">
21
+ {{ 'Connected' if status.get('data', {}).get('connected') else 'Disconnected' }}
22
+ </span>
23
+ </div>
24
+ <div class="flex justify-between items-center">
25
+ <span>Session:</span>
26
+ <span id="session-status" class="font-medium">
27
+ {{ 'Active' if status.get('data', {}).get('has_session') else 'None' }}
28
+ </span>
29
+ </div>
30
+ <div class="flex justify-between items-center">
31
+ <span>Last Check:</span>
32
+ <span id="last-check" class="text-sm text-gray-600">
33
+ Never
34
+ </span>
35
+ </div>
36
+ </div>
37
+
38
+ <div class="mt-6 space-y-3">
39
+ <button id="start-btn" onclick="startWhatsApp()"
40
+ class="w-full bg-green-600 text-white py-2 px-4 rounded-lg hover:bg-green-700 transition">
41
+ Start WhatsApp
42
+ </button>
43
+ <button id="restart-btn" onclick="restartWhatsApp()"
44
+ class="w-full bg-orange-600 text-white py-2 px-4 rounded-lg hover:bg-orange-700 transition">
45
+ Restart
46
+ </button>
47
+ </div>
48
+ </div>
49
+
50
+ <!-- QR Code Card -->
51
+ <div class="bg-white rounded-lg shadow-md p-6">
52
+ <h2 class="text-xl font-semibold mb-4 flex items-center">
53
+ <svg class="w-6 h-6 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
54
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4.354a4 4 0 110 5.292M15 21H3v-1a6 6 0 0112 0v1zm0 0h6v-1a6 6 0 00-9-5.197m13.5-9a2.5 2.5 0 11-5 0 2.5 2.5 0 015 0z"></path>
55
+ </svg>
56
+ QR Code
57
+ </h2>
58
+
59
+ <div id="qr-container" class="text-center">
60
+ <div class="bg-gray-100 rounded-lg p-8 mb-4">
61
+ <svg class="w-16 h-16 mx-auto text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
62
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4.354a4 4 0 110 5.292M15 21H3v-1a6 6 0 0112 0v1zm0 0h6v-1a6 6 0 00-9-5.197m13.5-9a2.5 2.5 0 11-5 0 2.5 2.5 0 015 0z"></path>
63
+ </svg>
64
+ <p class="text-gray-600 mt-2">No QR code available</p>
65
+ </div>
66
+ <p class="text-sm text-gray-600">
67
+ Start WhatsApp service to generate QR code for pairing
68
+ </p>
69
+ </div>
70
+ </div>
71
+
72
+ <!-- Quick Send Card -->
73
+ <div class="bg-white rounded-lg shadow-md p-6">
74
+ <h2 class="text-xl font-semibold mb-4 flex items-center">
75
+ <svg class="w-6 h-6 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
76
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 19l9 2-9-18-9 18 9-2zm0 0v-8"></path>
77
+ </svg>
78
+ Quick Send
79
+ </h2>
80
+
81
+ <form id="quick-send-form" class="space-y-4">
82
+ <div>
83
+ <label class="block text-sm font-medium mb-1">Phone Number</label>
84
+ <input type="text" id="quick-phone" placeholder="1234567890" required
85
+ class="w-full border rounded-lg px-3 py-2 focus:ring-2 focus:ring-green-500 focus:border-transparent">
86
+ </div>
87
+ <div>
88
+ <label class="block text-sm font-medium mb-1">Message</label>
89
+ <textarea id="quick-message" placeholder="Enter your message..." rows="3" required
90
+ class="w-full border rounded-lg px-3 py-2 focus:ring-2 focus:ring-green-500 focus:border-transparent"></textarea>
91
+ </div>
92
+ <button type="submit"
93
+ class="w-full bg-green-600 text-white py-2 px-4 rounded-lg hover:bg-green-700 transition">
94
+ Send Message
95
+ </button>
96
+ </form>
97
+ </div>
98
+ </div>
99
+
100
+ <!-- Activity Feed -->
101
+ <div class="mt-8 bg-white rounded-lg shadow-md p-6">
102
+ <h2 class="text-xl font-semibold mb-4">Activity Feed</h2>
103
+ <div id="activity-feed" class="space-y-2">
104
+ <p class="text-gray-600">No recent activity</p>
105
+ </div>
106
+ </div>
107
+ {% endblock %}
108
+
109
+ {% block scripts %}
110
+ <script>
111
+ // Dashboard specific JavaScript
112
+ let activityFeed = [];
113
+
114
+ function startWhatsApp() {
115
+ const btn = document.getElementById('start-btn');
116
+ btn.disabled = true;
117
+ btn.textContent = 'Starting...';
118
+
119
+ fetch('/api/start', { method: 'POST' })
120
+ .then(response => response.json())
121
+ .then(data => {
122
+ if (data.success) {
123
+ addActivity('success', 'WhatsApp service started successfully');
124
+ // Start loading QR code immediately
125
+ setTimeout(() => loadQRCode(), 1500);
126
+ } else {
127
+ addActivity('error', `Failed to start WhatsApp: ${data.error}`);
128
+ }
129
+ })
130
+ .catch(error => {
131
+ addActivity('error', `Error: ${error.message}`);
132
+ })
133
+ .finally(() => {
134
+ btn.disabled = false;
135
+ btn.textContent = 'Start WhatsApp';
136
+ setTimeout(refreshStatus, 2000);
137
+ });
138
+ }
139
+
140
+ function refreshStatus() {
141
+ fetch('/api/status')
142
+ .then(response => response.json())
143
+ .then(data => {
144
+ if (data.success && data.data) {
145
+ updateStatusDisplay(data.data);
146
+ addActivity('info', 'Status refreshed');
147
+ }
148
+ })
149
+ .catch(error => {
150
+ addActivity('error', `Failed to refresh status: ${error.message}`);
151
+ });
152
+ }
153
+
154
+ function updateStatusDisplay(status) {
155
+ document.getElementById('connection-status').textContent =
156
+ status.connected ? 'Connected' : 'Disconnected';
157
+ document.getElementById('session-status').textContent =
158
+ status.has_session ? 'Active' : 'None';
159
+ document.getElementById('last-check').textContent =
160
+ new Date().toLocaleTimeString();
161
+ }
162
+
163
+ function addActivity(type, message) {
164
+ const timestamp = new Date().toLocaleTimeString();
165
+ activityFeed.unshift({ type, message, timestamp });
166
+
167
+ // Keep only last 10 activities
168
+ if (activityFeed.length > 10) {
169
+ activityFeed = activityFeed.slice(0, 10);
170
+ }
171
+
172
+ renderActivityFeed();
173
+ }
174
+
175
+ function renderActivityFeed() {
176
+ const container = document.getElementById('activity-feed');
177
+ if (activityFeed.length === 0) {
178
+ container.innerHTML = '<p class="text-gray-600">No recent activity</p>';
179
+ return;
180
+ }
181
+
182
+ container.innerHTML = activityFeed.map(activity => {
183
+ const colorClass = activity.type === 'success' ? 'text-green-600' :
184
+ activity.type === 'error' ? 'text-red-600' : 'text-blue-600';
185
+ return `
186
+ <div class="flex justify-between items-center py-2 border-b last:border-b-0">
187
+ <span class="${colorClass}">${activity.message}</span>
188
+ <span class="text-sm text-gray-500">${activity.timestamp}</span>
189
+ </div>
190
+ `;
191
+ }).join('');
192
+ }
193
+
194
+ // Quick send form
195
+ document.getElementById('quick-send-form').addEventListener('submit', function(e) {
196
+ e.preventDefault();
197
+
198
+ const phone = document.getElementById('quick-phone').value;
199
+ const message = document.getElementById('quick-message').value;
200
+
201
+ if (!phone || !message) {
202
+ addActivity('error', 'Phone number and message are required');
203
+ return;
204
+ }
205
+
206
+ fetch('/api/send', {
207
+ method: 'POST',
208
+ headers: { 'Content-Type': 'application/json' },
209
+ body: JSON.stringify({ phone, message })
210
+ })
211
+ .then(response => response.json())
212
+ .then(data => {
213
+ if (data.success) {
214
+ addActivity('success', `Message sent to ${phone}`);
215
+ document.getElementById('quick-message').value = '';
216
+ } else {
217
+ addActivity('error', `Failed to send message: ${data.error}`);
218
+ }
219
+ })
220
+ .catch(error => {
221
+ addActivity('error', `Error sending message: ${error.message}`);
222
+ });
223
+ });
224
+
225
+ function loadQRCode(maxAttempts = 10, attemptNum = 1) {
226
+ fetch('/api/qr')
227
+ .then(response => response.json())
228
+ .then(data => {
229
+ const container = document.getElementById('qr-container');
230
+ if (data.success && data.data && data.data.has_qr) {
231
+ // Display base64 image directly (Docker compatible)
232
+ const imgSrc = data.data.image_data
233
+ ? `data:image/png;base64,${data.data.image_data}`
234
+ : `/qr/${data.data.filename}`;
235
+ container.innerHTML = `
236
+ <div class="text-center">
237
+ <img src="${imgSrc}" alt="QR Code" class="mx-auto mb-4 border-2 border-gray-300 rounded" style="width: 256px; height: 256px;" />
238
+ <p class="text-sm text-gray-600">Scan this QR code with WhatsApp</p>
239
+ </div>
240
+ `;
241
+ addActivity('info', 'QR code loaded');
242
+ } else {
243
+ // QR not ready yet, poll again if we haven't exceeded max attempts
244
+ if (attemptNum < maxAttempts) {
245
+ container.innerHTML = `
246
+ <div class="bg-gray-100 rounded-lg p-8 mb-4">
247
+ <svg class="w-16 h-16 mx-auto text-gray-400 animate-spin" fill="none" stroke="currentColor" viewBox="0 0 24 24">
248
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"></path>
249
+ </svg>
250
+ <p class="text-gray-600 mt-2">Waiting for QR code... (${attemptNum}/${maxAttempts})</p>
251
+ </div>
252
+ `;
253
+ setTimeout(() => loadQRCode(maxAttempts, attemptNum + 1), 1000);
254
+ } else {
255
+ container.innerHTML = `
256
+ <div class="bg-gray-100 rounded-lg p-8 mb-4">
257
+ <svg class="w-16 h-16 mx-auto text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
258
+ <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4.354a4 4 0 110 5.292M15 21H3v-1a6 6 0 0112 0v1zm0 0h6v-1a6 6 0 00-9-5.197m13.5-9a2.5 2.5 0 11-5 0 2.5 2.5 0 015 0z"></path>
259
+ </svg>
260
+ <p class="text-gray-600 mt-2">QR code not available</p>
261
+ </div>
262
+ <p class="text-sm text-gray-600">
263
+ Try clicking "Regenerate QR Code" again
264
+ </p>
265
+ `;
266
+ addActivity('error', 'QR code generation timed out');
267
+ }
268
+ }
269
+ })
270
+ .catch(error => {
271
+ addActivity('error', `Failed to load QR code: ${error.message}`);
272
+ });
273
+ }
274
+
275
+ function restartWhatsApp() {
276
+ const btn = event.target;
277
+ btn.disabled = true;
278
+ btn.textContent = 'Restarting...';
279
+
280
+ fetch('/api/restart', { method: 'POST' })
281
+ .then(response => response.json())
282
+ .then(data => {
283
+ if (data.success) {
284
+ addActivity('success', 'WhatsApp service restarted successfully');
285
+ // Start loading QR code immediately
286
+ setTimeout(() => loadQRCode(), 1500);
287
+ } else {
288
+ addActivity('error', `Failed to restart WhatsApp: ${data.error}`);
289
+ }
290
+ })
291
+ .catch(error => {
292
+ addActivity('error', `Error: ${error.message}`);
293
+ })
294
+ .finally(() => {
295
+ btn.disabled = false;
296
+ btn.textContent = 'Restart';
297
+ setTimeout(refreshStatus, 2000);
298
+ });
299
+ }
300
+
301
+ // Socket.IO events for real-time updates
302
+ socket.on('qr_code', function(data) {
303
+ loadQRCode(); // Reload QR code when event received
304
+ addActivity('info', 'New QR code generated');
305
+ });
306
+
307
+ socket.on('connected', function(data) {
308
+ addActivity('success', 'WhatsApp connected successfully');
309
+ refreshStatus();
310
+ });
311
+
312
+ socket.on('disconnected', function(data) {
313
+ addActivity('warning', 'WhatsApp disconnected');
314
+ refreshStatus();
315
+ });
316
+
317
+ // Initial status check only (QR code must be manually regenerated)
318
+ refreshStatus();
319
+ </script>
320
+ {% endblock %}