roksta 0.2.7__cp312-cp312-macosx_10_13_universal2.whl → 0.3.2__cp312-cp312-macosx_10_13_universal2.whl

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.

Potentially problematic release.


This version of roksta might be problematic. Click here for more details.

Files changed (91) hide show
  1. roksta/ai/call_ai.cpython-312-darwin.so +0 -0
  2. roksta/ai/gemini.cpython-312-darwin.so +0 -0
  3. roksta/ai/generic.cpython-312-darwin.so +0 -0
  4. roksta/ai/llm.cpython-312-darwin.so +0 -0
  5. roksta/ai/openai.cpython-312-darwin.so +0 -0
  6. roksta/ai/tools/__init__.cpython-312-darwin.so +0 -0
  7. roksta/ai/tools/delete_file.cpython-312-darwin.so +0 -0
  8. roksta/ai/tools/edit_file.cpython-312-darwin.so +0 -0
  9. roksta/ai/tools/final_response.cpython-312-darwin.so +0 -0
  10. roksta/ai/tools/get_file_summaries.cpython-312-darwin.so +0 -0
  11. roksta/ai/tools/read_file.cpython-312-darwin.so +0 -0
  12. roksta/ai/tools/regex_replace.cpython-312-darwin.so +0 -0
  13. roksta/ai/tools/shell_any.cpython-312-darwin.so +0 -0
  14. roksta/ai/tools/shell_limited.cpython-312-darwin.so +0 -0
  15. roksta/ai/tools/tool_defs.cpython-312-darwin.so +0 -0
  16. roksta/ai/tools/tool_utils.cpython-312-darwin.so +0 -0
  17. roksta/ai/tools/write_file.cpython-312-darwin.so +0 -0
  18. roksta/balance.cpython-312-darwin.so +0 -0
  19. roksta/check_for_updates.cpython-312-darwin.so +0 -0
  20. roksta/clarify_goal.cpython-312-darwin.so +0 -0
  21. roksta/command_handlers/__init__.cpython-312-darwin.so +0 -0
  22. roksta/command_handlers/handle_activate_command.cpython-312-darwin.so +0 -0
  23. roksta/command_handlers/handle_add_funds_command.cpython-312-darwin.so +0 -0
  24. roksta/command_handlers/handle_auto_charge_command.cpython-312-darwin.so +0 -0
  25. roksta/command_handlers/handle_auto_commit_command.cpython-312-darwin.so +0 -0
  26. roksta/command_handlers/handle_building_command.cpython-312-darwin.so +0 -0
  27. roksta/command_handlers/handle_chat_command.cpython-312-darwin.so +0 -0
  28. roksta/command_handlers/handle_dev_rate_command.cpython-312-darwin.so +0 -0
  29. roksta/command_handlers/handle_feedback_command.cpython-312-darwin.so +0 -0
  30. roksta/command_handlers/handle_goal_command.cpython-312-darwin.so +0 -0
  31. roksta/command_handlers/handle_help_command.cpython-312-darwin.so +0 -0
  32. roksta/command_handlers/handle_init_command.cpython-312-darwin.so +0 -0
  33. roksta/command_handlers/handle_linting_command.cpython-312-darwin.so +0 -0
  34. roksta/command_handlers/handle_login_command.cpython-312-darwin.so +0 -0
  35. roksta/command_handlers/handle_logout_command.cpython-312-darwin.so +0 -0
  36. roksta/command_handlers/handle_payment_details_command.cpython-312-darwin.so +0 -0
  37. roksta/command_handlers/handle_quit_command.cpython-312-darwin.so +0 -0
  38. roksta/command_handlers/handle_redeem_command.cpython-312-darwin.so +0 -0
  39. roksta/command_handlers/handle_request_activation_command.cpython-312-darwin.so +0 -0
  40. roksta/command_handlers/handle_testing_command.cpython-312-darwin.so +0 -0
  41. roksta/command_handlers/handle_usage_command.cpython-312-darwin.so +0 -0
  42. roksta/create_default_config.cpython-312-darwin.so +0 -0
  43. roksta/env.cpython-312-darwin.so +0 -0
  44. roksta/extended_text_area.cpython-312-darwin.so +0 -0
  45. roksta/firebase.cpython-312-darwin.so +0 -0
  46. roksta/fix_tests.cpython-312-darwin.so +0 -0
  47. roksta/goal_workflow.cpython-312-darwin.so +0 -0
  48. roksta/init_codebase.cpython-312-darwin.so +0 -0
  49. roksta/lint_code.cpython-312-darwin.so +0 -0
  50. roksta/main.cpython-312-darwin.so +0 -0
  51. roksta/new_features.cpython-312-darwin.so +0 -0
  52. roksta/propose_solution.cpython-312-darwin.so +0 -0
  53. roksta/roksta.cpython-312-darwin.so +0 -0
  54. roksta/select_files.cpython-312-darwin.so +0 -0
  55. roksta/tips.cpython-312-darwin.so +0 -0
  56. roksta/utils.cpython-312-darwin.so +0 -0
  57. roksta/write_code.cpython-312-darwin.so +0 -0
  58. {roksta-0.2.7.dist-info → roksta-0.3.2.dist-info}/METADATA +2 -1
  59. roksta-0.3.2.dist-info/RECORD +121 -0
  60. tests/conftest.py +42 -0
  61. tests/functions/{api_v0_01 → api_v1_01}/__init__.py +1 -1
  62. tests/functions/{api_v0_01 → api_v1_01}/test__analytics.py +2 -3
  63. tests/functions/{api_v0_01 → api_v1_01}/test__gemini_proxy.py +51 -6
  64. tests/functions/{api_v0_01 → api_v1_01}/test__generic_proxy.py +31 -2
  65. tests/functions/{api_v0_01 → api_v1_01}/test__get_payment_details.py +2 -2
  66. tests/functions/{api_v0_01 → api_v1_01}/test__openai_proxy.py +50 -14
  67. tests/functions/{api_v0_01 → api_v1_01}/test__redeem_credit_code.py +2 -2
  68. tests/functions/{api_v0_01 → api_v1_01}/test__sync_emails.py +3 -2
  69. tests/functions/{api_v0_01 → api_v1_01}/test__take_payment.py +2 -2
  70. tests/functions/{api_v0_01 → api_v1_01}/test__use_activation_code.py +3 -2
  71. tests/functions/api_v1_02/__init__.py +2 -0
  72. tests/functions/api_v1_02/test__analytics.py +416 -0
  73. tests/functions/api_v1_02/test__gemini_proxy.py +352 -0
  74. tests/functions/api_v1_02/test__generic_proxy.py +428 -0
  75. tests/functions/api_v1_02/test__get_payment_details.py +356 -0
  76. tests/functions/api_v1_02/test__openai_proxy.py +449 -0
  77. tests/functions/api_v1_02/test__redeem_credit_code.py +167 -0
  78. tests/functions/api_v1_02/test__sync_emails.py +325 -0
  79. tests/functions/api_v1_02/test__take_payment.py +491 -0
  80. tests/functions/api_v1_02/test__use_activation_code.py +438 -0
  81. tests/functions/api_v1_02/test_proxy_keyword_replacement.py +557 -0
  82. tests/functions/api_v1_02/test_replace_keywords.py +74 -0
  83. tests/functions/test_utils.py +484 -0
  84. roksta/ai/tools.cpython-312-darwin.so +0 -0
  85. roksta/command_handlers.cpython-312-darwin.so +0 -0
  86. roksta-0.2.7.dist-info/RECORD +0 -78
  87. tests/functions/test_utils_functions.py +0 -222
  88. {roksta-0.2.7.dist-info → roksta-0.3.2.dist-info}/WHEEL +0 -0
  89. {roksta-0.2.7.dist-info → roksta-0.3.2.dist-info}/entry_points.txt +0 -0
  90. {roksta-0.2.7.dist-info → roksta-0.3.2.dist-info}/top_level.txt +0 -0
  91. /tests/functions/{test_main_functions.py → test_main.py} +0 -0
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: roksta
3
- Version: 0.2.7
3
+ Version: 0.3.2
4
4
  Summary: An AI coding assistant
5
5
  Author-email: Prash Naidu <prash@roksta.ai>
6
6
  Project-URL: Homepage, https://roksta.ai
@@ -21,6 +21,7 @@ Requires-Dist: gitpython>=3.1.44
21
21
  Requires-Dist: jsonref>=1.1.0
22
22
  Requires-Dist: pathspec>=0.12.1
23
23
  Requires-Dist: psutil>=7.1.0
24
+ Requires-Dist: PyNaCl>=1.6.0
24
25
  Provides-Extra: tests
25
26
  Requires-Dist: pytest>=8.4.1; extra == "tests"
26
27
  Requires-Dist: pytest-asyncio>=1.0.0; extra == "tests"
@@ -0,0 +1,121 @@
1
+ roksta-0.3.2.dist-info/RECORD,,
2
+ roksta-0.3.2.dist-info/WHEEL,sha256=3DzZhK-rNIkkEIppB3qbu6syT4kayeV5HkusnWIqMIg,142
3
+ roksta-0.3.2.dist-info/entry_points.txt,sha256=mzRdYg_DlzZRwjxYUt9-gyoRCkM1QBTeTbwETgiTdGw,44
4
+ roksta-0.3.2.dist-info/top_level.txt,sha256=lvciNZQ1dPGXpiCLdWVXK03n9fKHjbQdwjqQbnUjeYM,13
5
+ roksta-0.3.2.dist-info/METADATA,sha256=XgnSDc8MqytQD1jnmYC-tuyMdYvoa0MPRR_JP-ugZbg,1456
6
+ roksta/default_config.cpython-312-darwin.so,sha256=Y-DEmdi-YpyroffYtiQ631HjKd-bZVWRxn49LIDRUcc,102232
7
+ roksta/__init__.cpython-312-darwin.so,sha256=aGhoZOTGADw_8ui447BG2io1Nf9nx4Qs353LjHeBrFc,85632
8
+ roksta/codebase_listing.cpython-312-darwin.so,sha256=GDTBAN99YQvxlb6RTKaBN3I7wn-7zc2jVlDdIaGpGVE,155720
9
+ roksta/make_issue.cpython-312-darwin.so,sha256=Sjvsc3d8Tkvkklm2n8VXymWeFfJGB6Sf8XUDLF8K-L8,221968
10
+ roksta/create_default_config.cpython-312-darwin.so,sha256=oMkNPhyY3B6Kb6Uq9b-a0S6aFrnb-epCkAr48qtSWuM,121864
11
+ roksta/gen_codebase_summaries.cpython-312-darwin.so,sha256=2AJ-0gS93G9jk8Zr9iH0QVbknSijFnbjhtQdEULYVrw,637776
12
+ roksta/balance.cpython-312-darwin.so,sha256=OKU-VXCqv41-jxCRrxfSKATIh9ONIgO5QLOLUpG5IcU,373216
13
+ roksta/propose_solution.cpython-312-darwin.so,sha256=r5vAICQyNZLwf_bkhXx54Ie6SPFjfw1Q-yCOQc0ppO8,190968
14
+ roksta/analytics.cpython-312-darwin.so,sha256=LLY7YZ95KCkyE-3OpYx04fE0OsPsv7PitQNhjBWvnLk,273024
15
+ roksta/utils.cpython-312-darwin.so,sha256=1e3ty9yN7-NxYwhYYpX9nLN42qldx5MORwGe6NXIW7E,621816
16
+ roksta/check_for_updates.cpython-312-darwin.so,sha256=5E1TvteXMzHUUF7BFCYJ2q0kZy6fBkDArZAV3Qf_PnM,306632
17
+ roksta/response_formats.cpython-312-darwin.so,sha256=nnb92jNdRh8g4Xq1ocI6xqeDvIKMX9x_1etlDcA-t8M,121112
18
+ roksta/main.cpython-312-darwin.so,sha256=dBhfkx0lHPe9a0SpIx73RhAbs6R-HPdHz0-EiobAVPA,138808
19
+ roksta/env.cpython-312-darwin.so,sha256=ourr06NyPEf5Hbgpskj0WvNkHunfHwpK3TtM8yvxFBo,121032
20
+ roksta/get_failing_tests.cpython-312-darwin.so,sha256=WRLjGOl5idN3jrSaC5b9L81VZTlCYMSnjwbuDUSemIg,224152
21
+ roksta/fix_tests.cpython-312-darwin.so,sha256=YftbgJPPsQL1HNaB324xEUZodVR7eC1inaIEm1V0CSA,224160
22
+ roksta/firebase_auth_web.cpython-312-darwin.so,sha256=UL94FIDhUwIt3befN3FmCyiPANxo_66dLb9Wp5CojR4,472600
23
+ roksta/run_cli_goal.cpython-312-darwin.so,sha256=ZMHes3egj1mezMkqHGRtGbVjCWf68rsGhQw1qt5NjS4,223808
24
+ roksta/build_project.cpython-312-darwin.so,sha256=R4p3o9ZxFNVcs1-y2ovIfdiHE5xryMLkv9YgSoyd1ks,191040
25
+ roksta/rewrite_goal.cpython-312-darwin.so,sha256=rIME3oOjnPK41PWYJaHQpyGuqgxIrt15ld-dNpqb8M4,190944
26
+ roksta/roksta.cpython-312-darwin.so,sha256=8dpBYVU3wJRxR0Dt-bXHjSdJvjZW2tDEKqlUyN44UCQ,571696
27
+ roksta/clarify_goal.cpython-312-darwin.so,sha256=Dpg8KzTpXoK89U5GGa9oSSHPWRSbvetum94AurttIGs,207680
28
+ roksta/init_codebase.cpython-312-darwin.so,sha256=Lhs3Sig-rhAyLm4QETiWBeMnNotRymEKpoKJuYMWYSc,341376
29
+ roksta/gen_one_line_goal.cpython-312-darwin.so,sha256=_7TixHwo32-8MFKj8tYln6wERC3mVkRC9oWfH09rgAk,207624
30
+ roksta/new_features.cpython-312-darwin.so,sha256=YS-7Fo_IjAMG0Mu6v7_uAP346rewn3a6OcRUF_pWJaY,85712
31
+ roksta/firebase_config.cpython-312-darwin.so,sha256=EOpQM7e9pW2ZqAlkjbeOV-Iyoq4DSU__DIZpeDmO9gI,86776
32
+ roksta/chat_workflow.cpython-312-darwin.so,sha256=DKC26wml81h4tNSE5-Bji5C92y2A94j3_Aayq_sbNOE,273664
33
+ roksta/select_files.cpython-312-darwin.so,sha256=5XVLIUuZvc_HEnWwfvrag4kwmYNcRfm7vowrl39aw6g,207648
34
+ roksta/enums.cpython-312-darwin.so,sha256=13gc6A4RbZG7DGQIk5o2PlRwX4ymKpdiOzcJV58rxFQ,338440
35
+ roksta/write_code.cpython-312-darwin.so,sha256=gKNlx4wJTJFmf5PtuwBgSB5wPuqK8qrBTzOTYz51L8Q,191008
36
+ roksta/tips.cpython-312-darwin.so,sha256=6agS5I-HPsEUYlMGdLb6SZgRQ3E4QXYtbIMw0SxQf7M,85672
37
+ roksta/goal_workflow.cpython-312-darwin.so,sha256=rcwqDYT07mNjA3FbU1kFux8xjejtycJ8pMiZGDp6FNE,521760
38
+ roksta/firebase.cpython-312-darwin.so,sha256=WAl2UK3th8Xw7ZKbPravorAnqTCHsJkOt5GQaxzrccU,488048
39
+ roksta/parse_readme.cpython-312-darwin.so,sha256=UQaLbWoB2rOPPDMHSNDA9muAptft1iJ-vkjyKWN9T4s,207152
40
+ roksta/logger.cpython-312-darwin.so,sha256=lIfX2V-QExvKZK3LcLF4qQGeMduT5ZVPUXwg0kk57rU,122160
41
+ roksta/lint_code.cpython-312-darwin.so,sha256=QgEkYdj4J5MZF3IyCn9M69H-XZ5ihNiODVaw9TZkvpU,224496
42
+ roksta/extended_text_area.cpython-312-darwin.so,sha256=uSOqAGbFrgI0jkwWNY636WOfMTc9mdzXs3qT_WdwDLM,307272
43
+ roksta/checkpoints.cpython-312-darwin.so,sha256=UdKnNkQweTrMxDjpfjruXtEFVyNG4B4mrg2K07f3ojg,254432
44
+ roksta/get_codebase_structure.cpython-312-darwin.so,sha256=_ZS1xgbWZUm8grv1lrmstrepFvH-ZK-xczOQ4-doM60,206272
45
+ roksta/command_handlers/__init__.cpython-312-darwin.so,sha256=w1SkWvlEUw7c0hKd9HY1_teX8cKcmdbYKUcWCMGWinw,86256
46
+ roksta/command_handlers/handle_logout_command.cpython-312-darwin.so,sha256=2l7y8vA4-3aquMJZBBuqDrzdEhBn2-O4obMHqVlR8ec,121624
47
+ roksta/command_handlers/handle_redeem_command.cpython-312-darwin.so,sha256=SU7Rn60su7bIn4ZjR_u5L_z8izk-cMlYQyOD-s1RRwE,157800
48
+ roksta/command_handlers/handle_usage_command.cpython-312-darwin.so,sha256=x9ZRZN4KeGSdVXrpTwtGotmDTNRLKxqJO7oqtVb5c3c,207944
49
+ roksta/command_handlers/handle_auto_charge_command.cpython-312-darwin.so,sha256=HNv6EISO2Uh-ZDohRYoUzVSkYgb22b4H7mb5PZIdJ6U,172944
50
+ roksta/command_handlers/handle_building_command.cpython-312-darwin.so,sha256=D9UHtfr1n604eR1oRfqMB66Ky7QlJbCV2urmO4IGl4c,156080
51
+ roksta/command_handlers/handle_auto_commit_command.cpython-312-darwin.so,sha256=5clPnjCSSIrZwXU_DI3fqmXp16cdwMKiMIWydtIsHI4,122672
52
+ roksta/command_handlers/handle_help_command.cpython-312-darwin.so,sha256=1UcckeeCMdBSz_-IXzQ-i5w8zH2F4IjXKnILI3fBvTA,220568
53
+ roksta/command_handlers/handle_add_funds_command.cpython-312-darwin.so,sha256=JjSQAeGxaNSKm3JX3ffVdrpfmIenRgXuKOnQ1k6OJXE,224496
54
+ roksta/command_handlers/handle_chat_command.cpython-312-darwin.so,sha256=nZ2RsDXJHtLqXhUJ4GNBoQEcfRKX0-gZEGlrIf1E1mQ,156792
55
+ roksta/command_handlers/handle_testing_command.cpython-312-darwin.so,sha256=BeCYaWoYtu34UxGXttD68lFPP-vcm6zUJXODsJcGL1U,156064
56
+ roksta/command_handlers/handle_init_command.cpython-312-darwin.so,sha256=hdpTFH0aO2Ov6qWi5W6DcBKVA9rHlQwhNxb9TtpPP3o,157192
57
+ roksta/command_handlers/handle_dev_rate_command.cpython-312-darwin.so,sha256=AbEeAhX2Dlih9Y-MkS37wsQxZeOCpwlGfJseeurmE7Q,123168
58
+ roksta/command_handlers/handle_activate_command.cpython-312-darwin.so,sha256=7z1vtwp5XuTRDG_oATzKJpRY2CyNBA9nFPXVxeX7CNM,240464
59
+ roksta/command_handlers/handle_login_command.cpython-312-darwin.so,sha256=yMwfuLVvV4rcTFEOUvPtMXp6mmmOuACnICAKAP1sNRg,207240
60
+ roksta/command_handlers/handle_linting_command.cpython-312-darwin.so,sha256=quNAw6TV67G0Mfq2npeUcwcY5ljfqNBFnvVJt7jTyFk,156080
61
+ roksta/command_handlers/handle_quit_command.cpython-312-darwin.so,sha256=5e-Dl892KJu6wfOycgKSKyefcmgwXj4u17Oln0LvcO8,87656
62
+ roksta/command_handlers/handle_payment_details_command.cpython-312-darwin.so,sha256=UghdU6OL3hj7Cc_ozLLEesg3YbDq5i4JFHHByYZbwlU,173640
63
+ roksta/command_handlers/handle_goal_command.cpython-312-darwin.so,sha256=LkINQmNLpXB4Qi13inV4Zp42UYwajR60xpdkgSLFQFM,173576
64
+ roksta/command_handlers/handle_request_activation_command.cpython-312-darwin.so,sha256=5qF9R4BoQ2DDIqM5EAQ-EJIcxMjqwkSby8XHfXvLo6g,206968
65
+ roksta/command_handlers/handle_feedback_command.cpython-312-darwin.so,sha256=UwUgqvIgb3K1tVpCflNPtGa1wDGIzIQLuECAOwednfg,206736
66
+ roksta/ai/__init__.cpython-312-darwin.so,sha256=5_lHxLo2XMbdiUPvD567zM1pag2REJBiRZkorL9pEdw,85632
67
+ roksta/ai/generic.cpython-312-darwin.so,sha256=b8LmM5kjFBcliylSqLPbc4BRpsrOmqpG2f5N2RTOIaM,306800
68
+ roksta/ai/gemini.cpython-312-darwin.so,sha256=pIq0T2CgrIelSKOy9Mkh8_fFVmOv03VMwwV-oxrkvEM,406000
69
+ roksta/ai/openai.cpython-312-darwin.so,sha256=HFUNdQmK68gRBYc_dVEG6oHj9i55W2Yqv0iLGDbq4zU,290400
70
+ roksta/ai/call_ai.cpython-312-darwin.so,sha256=dFFRIydEtqcC6CrJwzTV5ERNeKXms5CHGoneOtanlyE,240624
71
+ roksta/ai/llm.cpython-312-darwin.so,sha256=YClogWSkgzQmV03Dkw3aYZf_VITWT1wKAYJbxebrSes,271640
72
+ roksta/ai/tools/__init__.cpython-312-darwin.so,sha256=yvKyTdurBPmXVk8dCzrOvyv5Br2bJxyHEgNuDZoYkcQ,87392
73
+ roksta/ai/tools/tool_defs.cpython-312-darwin.so,sha256=EIApR1AWapy9_4VcYXd3MliPssnyj-kncXa5PunKyGc,137520
74
+ roksta/ai/tools/tool_utils.cpython-312-darwin.so,sha256=5DHXLk9C4RZ-EbKbWNgJ5o9np9KiDIc3-XMlNhOVEr0,307584
75
+ roksta/ai/tools/final_response.cpython-312-darwin.so,sha256=cxaX6NjwRSmP6SabPW2Ye0lC2jxBsLt8sfcntg79XpQ,88088
76
+ roksta/ai/tools/regex_replace.cpython-312-darwin.so,sha256=2lSzL8Xrxqa9TIWM1CgBPygSIT6O6p-QdNphHbDaJUE,155616
77
+ roksta/ai/tools/shell_any.cpython-312-darwin.so,sha256=j4yFJa12aI7rQheAYyIb8zjkURNjO8x-rpGndXHxvtI,224064
78
+ roksta/ai/tools/shell_limited.cpython-312-darwin.so,sha256=YG6v8qGbnxFq_lo-BjFxku3O_34k3SOWolB-yRcSnms,190848
79
+ roksta/ai/tools/delete_file.cpython-312-darwin.so,sha256=GeDRmXhEuOngYkuk8xi6C2gXlTIDk7f_RG0oVPSZFco,122096
80
+ roksta/ai/tools/read_file.cpython-312-darwin.so,sha256=-MvzhPtU7WZUpvezwzP7QLLqaB9UpmMxVDD8_7pNq6I,155920
81
+ roksta/ai/tools/write_file.cpython-312-darwin.so,sha256=ymftllGtvtlRLyzs1sJtebKlhEibIJCafbGcdn_p0CE,122160
82
+ roksta/ai/tools/get_file_summaries.cpython-312-darwin.so,sha256=sBW_uHxfIUJLJHhY4ZYuPTUhpINdA6Q5cXGZMhRpsoE,122840
83
+ roksta/ai/tools/edit_file.cpython-312-darwin.so,sha256=zMDvLx_rXJvvhNEVl1ONjPpikavVdS_fg8ZeGETGveg,155280
84
+ tests/conftest.py,sha256=Y2cx3Jups9GkKGxOG2fSSoIr3rgfUa9QzipMoQ-ehKU,7878
85
+ tests/__init__.py,sha256=J6ztnXNqM8TSQCgHXMWb6NRq1kN_h4ql63Jq-TYR8XE,124
86
+ tests/functions/test_utils.py,sha256=_zlgSa99JYVyJ0LDe3yaPztS92kqGRKkgjon__XmfFM,16721
87
+ tests/functions/test_auth.py,sha256=I3oFTw2LTuUsnQ75v5SCfgUPnpel_pDKqUH0VAJpXQw,992
88
+ tests/functions/__init__.py,sha256=mLXx3UARcv07vs_fGJuy3L5ZRn9IRsF9zIuL5IOnwdk,95
89
+ tests/functions/test_main.py,sha256=h0W_4ISo95WzRy2HpXZw3cuNOTcYk_2QCIZffZJEfEU,2575
90
+ tests/functions/api_v1_02/test__sync_emails.py,sha256=OTyf2YbBK7e4pWIQTUGYR4x6il1Hb3g4FYqQ_fRj1nA,12699
91
+ tests/functions/api_v1_02/test__take_payment.py,sha256=I-XJasGuLW5wTMAu4nAgFkA8Mo_zVpGPvkJ2LrrJGCk,18022
92
+ tests/functions/api_v1_02/test_replace_keywords.py,sha256=pa7bZ8I9XgtI5dUZq-efcHh9amN2_XHP8abXYqtDGc0,2683
93
+ tests/functions/api_v1_02/test__gemini_proxy.py,sha256=mf8ubRmNxxxN80H6EA-iDPwu6Oue6Rmgxl6tiFNpfrU,13344
94
+ tests/functions/api_v1_02/test_proxy_keyword_replacement.py,sha256=I5H73_QydSKLiDp7hDHGHbSkDOdv-BT4gYHGqEiVtM0,23632
95
+ tests/functions/api_v1_02/test__analytics.py,sha256=oKjd7_zjnttcb-5BuA93yllkm0OOF4YWnfTUW1-icnM,15806
96
+ tests/functions/api_v1_02/__init__.py,sha256=3RCSoj_5UQPw9Z5NeAgYz2s4gfcZIi7g36qk_Ed5TiA,124
97
+ tests/functions/api_v1_02/test__use_activation_code.py,sha256=tNyTkkRtrpjei-7F31UNwrcU-PbP-QnkWNU7h2n5Yqo,16960
98
+ tests/functions/api_v1_02/test__redeem_credit_code.py,sha256=7GD3tGgO2Lk0BCjUmzZACJiwocEhLBvXrrut0x13IWU,6347
99
+ tests/functions/api_v1_02/test__openai_proxy.py,sha256=xlO2zmG64LlrKrbjFz-ds6Map7vy4SjIuHfZJDAqGK8,17726
100
+ tests/functions/api_v1_02/test__get_payment_details.py,sha256=i5Px3NkLxqw8D3tVPrYdRJJF5jlNmysSuX_NNolbIa4,13735
101
+ tests/functions/api_v1_02/test__generic_proxy.py,sha256=KZkOtmc71aOnWlz2L7B6CFhRNEV73bExG3GR14oD2L0,17673
102
+ tests/functions/api_v1_01/test__sync_emails.py,sha256=OTyf2YbBK7e4pWIQTUGYR4x6il1Hb3g4FYqQ_fRj1nA,12699
103
+ tests/functions/api_v1_01/test__take_payment.py,sha256=I-XJasGuLW5wTMAu4nAgFkA8Mo_zVpGPvkJ2LrrJGCk,18022
104
+ tests/functions/api_v1_01/test__gemini_proxy.py,sha256=mf8ubRmNxxxN80H6EA-iDPwu6Oue6Rmgxl6tiFNpfrU,13344
105
+ tests/functions/api_v1_01/test__analytics.py,sha256=oKjd7_zjnttcb-5BuA93yllkm0OOF4YWnfTUW1-icnM,15806
106
+ tests/functions/api_v1_01/__init__.py,sha256=3RCSoj_5UQPw9Z5NeAgYz2s4gfcZIi7g36qk_Ed5TiA,124
107
+ tests/functions/api_v1_01/test__use_activation_code.py,sha256=tNyTkkRtrpjei-7F31UNwrcU-PbP-QnkWNU7h2n5Yqo,16960
108
+ tests/functions/api_v1_01/test__redeem_credit_code.py,sha256=7GD3tGgO2Lk0BCjUmzZACJiwocEhLBvXrrut0x13IWU,6347
109
+ tests/functions/api_v1_01/test__openai_proxy.py,sha256=xlO2zmG64LlrKrbjFz-ds6Map7vy4SjIuHfZJDAqGK8,17726
110
+ tests/functions/api_v1_01/test__get_payment_details.py,sha256=i5Px3NkLxqw8D3tVPrYdRJJF5jlNmysSuX_NNolbIa4,13735
111
+ tests/functions/api_v1_01/test__generic_proxy.py,sha256=KZkOtmc71aOnWlz2L7B6CFhRNEV73bExG3GR14oD2L0,17673
112
+ tests/functions/api_v1_00/test__sync_emails.py,sha256=OTyf2YbBK7e4pWIQTUGYR4x6il1Hb3g4FYqQ_fRj1nA,12699
113
+ tests/functions/api_v1_00/test__take_payment.py,sha256=I-XJasGuLW5wTMAu4nAgFkA8Mo_zVpGPvkJ2LrrJGCk,18022
114
+ tests/functions/api_v1_00/test__gemini_proxy.py,sha256=mf8ubRmNxxxN80H6EA-iDPwu6Oue6Rmgxl6tiFNpfrU,13344
115
+ tests/functions/api_v1_00/test__analytics.py,sha256=oKjd7_zjnttcb-5BuA93yllkm0OOF4YWnfTUW1-icnM,15806
116
+ tests/functions/api_v1_00/__init__.py,sha256=3RCSoj_5UQPw9Z5NeAgYz2s4gfcZIi7g36qk_Ed5TiA,124
117
+ tests/functions/api_v1_00/test__use_activation_code.py,sha256=tNyTkkRtrpjei-7F31UNwrcU-PbP-QnkWNU7h2n5Yqo,16960
118
+ tests/functions/api_v1_00/test__redeem_credit_code.py,sha256=7GD3tGgO2Lk0BCjUmzZACJiwocEhLBvXrrut0x13IWU,6347
119
+ tests/functions/api_v1_00/test__openai_proxy.py,sha256=xlO2zmG64LlrKrbjFz-ds6Map7vy4SjIuHfZJDAqGK8,17726
120
+ tests/functions/api_v1_00/test__get_payment_details.py,sha256=i5Px3NkLxqw8D3tVPrYdRJJF5jlNmysSuX_NNolbIa4,13735
121
+ tests/functions/api_v1_00/test__generic_proxy.py,sha256=KZkOtmc71aOnWlz2L7B6CFhRNEV73bExG3GR14oD2L0,17673
tests/conftest.py CHANGED
@@ -167,3 +167,45 @@ except Exception:
167
167
  _attach_gtypes_stubs(gtypes)
168
168
  genai.types = gtypes
169
169
  sys.modules['google.genai.types'] = gtypes
170
+
171
+ # Provide a minimal nacl.public stub if PyNaCl is not installed. This prevents
172
+ # import-time failures in tests that import `roksta.firebase` which does
173
+ # `from nacl.public import PublicKey, SealedBox`.
174
+ try:
175
+ import nacl.public # type: ignore
176
+ except Exception:
177
+ # Create lightweight stub modules to satisfy `from nacl.public import PublicKey, SealedBox`
178
+ nacl = _types_mod.ModuleType('nacl')
179
+ nacl_public = _types_mod.ModuleType('nacl.public')
180
+
181
+ class PublicKey:
182
+ def __init__(self, data):
183
+ # Accept bytes-like input; ensure it's bytes when possible.
184
+ try:
185
+ self._data = bytes(data)
186
+ except Exception:
187
+ self._data = data
188
+
189
+ def __repr__(self):
190
+ try:
191
+ length = len(self._data)
192
+ except Exception:
193
+ length = 'unknown'
194
+ return f"<PublicKey len={length}>"
195
+
196
+ class SealedBox:
197
+ def __init__(self, public_key):
198
+ self.public_key = public_key
199
+
200
+ def encrypt(self, plaintext: bytes) -> bytes:
201
+ # Return deterministic bytes so callers can base64-encode the result.
202
+ if not isinstance(plaintext, (bytes, bytearray)):
203
+ plaintext = str(plaintext).encode('utf-8')
204
+ return b"STUB_ENCRYPTED:" + bytes(plaintext)
205
+
206
+ nacl_public.PublicKey = PublicKey
207
+ nacl_public.SealedBox = SealedBox
208
+ # Attach the submodule on the parent module object for attribute access
209
+ nacl.public = nacl_public
210
+ sys.modules['nacl'] = nacl
211
+ sys.modules['nacl.public'] = nacl_public
@@ -1,2 +1,2 @@
1
- # tests.functions.api_v0_01 package marker
1
+ # tests.functions.api_v1_00 package marker
2
2
  # Allows tests under this directory to be imported with package-qualified names.
@@ -176,12 +176,11 @@ sys.modules['httpx'] = httpx_mod
176
176
  # -----------------------------
177
177
  repo_root = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..', '..'))
178
178
  functions_root = os.path.join(repo_root, 'functions')
179
- module_path = os.path.join(functions_root, 'api_v0_01', '_analytics.py')
180
- spec = importlib.util.spec_from_file_location('api_v0_01._analytics', module_path)
179
+ module_path = os.path.join(functions_root, 'api_v1_00', '_analytics.py')
180
+ spec = importlib.util.spec_from_file_location('api_v1_00._analytics', module_path)
181
181
  _analytics = importlib.util.module_from_spec(spec)
182
182
  spec.loader.exec_module(_analytics)
183
183
 
184
-
185
184
  # Restore original sys.modules mappings to avoid side-effects for other tests
186
185
  for name, orig in _orig_sys_modules.items():
187
186
  if orig is None:
@@ -19,6 +19,8 @@ if FUNCTIONS_DIR not in sys.path:
19
19
  _orig_sys_modules = {}
20
20
  _names_to_fake = [
21
21
  'firebase_functions',
22
+ 'firebase_admin',
23
+ 'firebase_admin.firestore',
22
24
  'utils',
23
25
  'auth',
24
26
  'google',
@@ -49,6 +51,35 @@ class FakeResponse:
49
51
  firebase_functions.https_fn = types.SimpleNamespace(Request=object, Response=FakeResponse)
50
52
  sys.modules['firebase_functions'] = firebase_functions
51
53
 
54
+ # Fake firebase_admin and its firestore submodule to satisfy optional imports in the proxy
55
+ firebase_admin = types.ModuleType('firebase_admin')
56
+ firestore_mod = types.ModuleType('firebase_admin.firestore')
57
+
58
+ # Minimal Client type for annotations and a no-op client() factory
59
+ class _DummyClient: # pragma: no cover - just a stub for import-time type hints
60
+ pass
61
+
62
+ def _dummy_client():
63
+ # Return a simple object; logic using it is patched in tests
64
+ return types.SimpleNamespace()
65
+
66
+ # Decorator stub used by perform_atomic_debit; never actually invoked in tests
67
+ def _transactional(fn=None):
68
+ def wrapper(*args, **kwargs):
69
+ return fn(*args, **kwargs)
70
+ return wrapper
71
+
72
+ # Populate stubs on the firestore module
73
+ setattr(firestore_mod, 'Client', _DummyClient)
74
+ setattr(firestore_mod, 'client', _dummy_client)
75
+ setattr(firestore_mod, 'transactional', _transactional)
76
+ setattr(firestore_mod, 'SERVER_TIMESTAMP', object())
77
+
78
+ # Attach firestore submodule to firebase_admin package and register in sys.modules
79
+ setattr(firebase_admin, 'firestore', firestore_mod)
80
+ sys.modules['firebase_admin'] = firebase_admin
81
+ sys.modules['firebase_admin.firestore'] = firestore_mod
82
+
52
83
  # Fake utils module (provides functions imported by _gemini_proxy)
53
84
  utils_mod = types.ModuleType('utils')
54
85
 
@@ -118,14 +149,14 @@ sys.modules['google.genai'] = genai_mod
118
149
  sys.modules['google.genai.types'] = types_mod
119
150
 
120
151
  # Import the module under test after preparing the fake imports
152
+ # Resolve the actual repository 'functions' directory (tests files live under tests/, so climb up to repo root)
121
153
  repo_root = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..', '..'))
122
154
  functions_root = os.path.join(repo_root, 'functions')
123
- module_path = os.path.join(functions_root, 'api_v0_01', '_gemini_proxy.py')
124
- spec = importlib.util.spec_from_file_location('api_v0_01._gemini_proxy', module_path)
155
+ module_path = os.path.join(functions_root, 'api_v1_00', '_gemini_proxy.py')
156
+ spec = importlib.util.spec_from_file_location('api_v1_00._gemini_proxy', module_path)
125
157
  _gemini = importlib.util.module_from_spec(spec)
126
158
  spec.loader.exec_module(_gemini)
127
159
 
128
-
129
160
  # Restore original sys.modules mappings to avoid side-effects for other tests
130
161
  for name, orig in _orig_sys_modules.items():
131
162
  if orig is None:
@@ -198,7 +229,9 @@ def test_invalid_auth_key_returns_403():
198
229
  def test_non_post_method_returns_405():
199
230
  headers = {_gemini.AUTH_HEADER_NAME: 'ok'}
200
231
  req = DummyRequest(headers=headers, method='GET')
201
- with patch.object(_gemini, 'validate_auth_key', return_value=True), patch.object(_gemini, 'verify_firebase_token', return_value={}):
232
+ with patch.object(_gemini, 'validate_auth_key', return_value=True), \
233
+ patch.object(_gemini, 'verify_firebase_token', return_value={}), \
234
+ patch.object(_gemini, 'ensure_balance_positive', return_value=(True, 100.0)):
202
235
  resp = _gemini._gemini_proxy(req)
203
236
 
204
237
  assert resp.status_code == 405
@@ -210,7 +243,9 @@ def test_non_post_method_returns_405():
210
243
  def test_malformed_json_returns_400():
211
244
  headers = {_gemini.AUTH_HEADER_NAME: 'ok'}
212
245
  req = DummyRequest(headers=headers, method='POST', raise_on_get_json=True)
213
- with patch.object(_gemini, 'validate_auth_key', return_value=True), patch.object(_gemini, 'verify_firebase_token', return_value={}):
246
+ with patch.object(_gemini, 'validate_auth_key', return_value=True), \
247
+ patch.object(_gemini, 'verify_firebase_token', return_value={}), \
248
+ patch.object(_gemini, 'ensure_balance_positive', return_value=(True, 100.0)):
214
249
  resp = _gemini._gemini_proxy(req)
215
250
 
216
251
  assert resp.status_code == 400
@@ -223,7 +258,9 @@ def test_rehydrate_params_error_returns_400():
223
258
  headers = {_gemini.AUTH_HEADER_NAME: 'ok'}
224
259
  # missing 'model' to cause KeyError inside rehydrate_params
225
260
  req = DummyRequest(headers=headers, method='POST', json_data={'contents': 'x', 'config': {}})
226
- with patch.object(_gemini, 'validate_auth_key', return_value=True), patch.object(_gemini, 'verify_firebase_token', return_value={}):
261
+ with patch.object(_gemini, 'validate_auth_key', return_value=True), \
262
+ patch.object(_gemini, 'verify_firebase_token', return_value={}), \
263
+ patch.object(_gemini, 'ensure_balance_positive', return_value=(True, 100.0)):
227
264
  resp = _gemini._gemini_proxy(req)
228
265
 
229
266
  assert resp.status_code == 400
@@ -237,6 +274,7 @@ def test_get_api_key_failure_returns_500():
237
274
  req = DummyRequest(headers=headers, method='POST', json_data={'model': 'g', 'contents': 'hi', 'config': {}})
238
275
  with patch.object(_gemini, 'validate_auth_key', return_value=True), \
239
276
  patch.object(_gemini, 'verify_firebase_token', return_value={}), \
277
+ patch.object(_gemini, 'ensure_balance_positive', return_value=(True, 100.0)), \
240
278
  patch.object(_gemini, 'get_api_key', side_effect=Exception('boom')):
241
279
  resp = _gemini._gemini_proxy(req)
242
280
 
@@ -258,12 +296,18 @@ def test_successful_flow_calls_genai_and_returns_payload():
258
296
 
259
297
  def generate_content(self, **params):
260
298
  class FakeResp:
299
+ def __init__(self):
300
+ # include usage metadata expected by get_usage()
301
+ self.usage_metadata = types.SimpleNamespace(prompt_token_count=1, total_token_count=2)
302
+
261
303
  def to_json_dict(self_inner):
262
304
  return { 'result': 'ok', 'received_model': params.get('model') }
263
305
  return FakeResp()
264
306
 
265
307
  with patch.object(_gemini, 'validate_auth_key', return_value=True), \
266
308
  patch.object(_gemini, 'verify_firebase_token', return_value={}), \
309
+ patch.object(_gemini, 'ensure_balance_positive', return_value=(True, 100.0)), \
310
+ patch.object(_gemini, 'bill_with_retry', return_value=("ok", 99.5)), \
267
311
  patch.object(_gemini, 'get_api_key', return_value='GEMINI-KEY'), \
268
312
  patch.object(_gemini.genai, 'Client', FakeClient):
269
313
  resp = _gemini._gemini_proxy(req)
@@ -297,6 +341,7 @@ def test_genai_exception_with_status_code_is_returned_as_error_status():
297
341
 
298
342
  with patch.object(_gemini, 'validate_auth_key', return_value=True), \
299
343
  patch.object(_gemini, 'verify_firebase_token', return_value={}), \
344
+ patch.object(_gemini, 'ensure_balance_positive', return_value=(True, 100.0)), \
300
345
  patch.object(_gemini, 'get_api_key', return_value='GEMINI-KEY'), \
301
346
  patch.object(_gemini.genai, 'Client', ErrClient):
302
347
  resp = _gemini._gemini_proxy(req)
@@ -108,13 +108,42 @@ openai_mod.APIError = APIError
108
108
  sys.modules['openai'] = openai_mod
109
109
 
110
110
  # Import the module under test after preparing the fake imports
111
+ #_generic = importlib.import_module('_generic_proxy')
112
+
111
113
  repo_root = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..', '..'))
112
114
  functions_root = os.path.join(repo_root, 'functions')
113
- module_path = os.path.join(functions_root, 'api_v0_01', '_generic_proxy.py')
114
- spec = importlib.util.spec_from_file_location('api_v0_01._generic_proxy', module_path)
115
+ module_path = os.path.join(functions_root, 'api_v1_00', '_generic_proxy.py')
116
+ spec = importlib.util.spec_from_file_location('api_v1_00._generic_proxy', module_path)
115
117
  _generic = importlib.util.module_from_spec(spec)
116
118
  spec.loader.exec_module(_generic)
117
119
 
120
+ # Patch Firestore client and billing functions to avoid external dependencies during tests
121
+ # Provide a dummy Firestore client and make balance checks/billing no-ops.
122
+ _generic.firestore = types.SimpleNamespace(client=lambda: object())
123
+
124
+ def _fake_ensure_balance_positive(db, user_id):
125
+ return True, 100.0
126
+
127
+ def _fake_extract_usage(llm_family=None, call_type=None, model_id=None, request_payload=None, response_payload=None):
128
+ # Default to zero-usage unless present in the payload
129
+ usage = response_payload.get('usage', {}) if isinstance(response_payload, dict) else {}
130
+ in_tokens = usage.get('prompt_tokens', 0) or usage.get('input_tokens', 0) or 0
131
+ out_tokens = usage.get('completion_tokens', 0) or usage.get('output_tokens', 0) or 0
132
+ return {"input_tokens": in_tokens, "output_tokens": out_tokens}
133
+
134
+ def _fake_calculate_cost(model_id=None, input_tokens=0, output_tokens=0):
135
+ return 0.0
136
+
137
+ def _fake_bill_with_retry(db=None, user_id=None, model_id=None, usage=None, cost=None, reason=None):
138
+ return "ok", 100.0
139
+
140
+ _generic.ensure_balance_positive = _fake_ensure_balance_positive
141
+ _generic.extract_usage = _fake_extract_usage
142
+ # Compatibility shim: some versions of the module call get_usage(response) instead of extract_usage(...)
143
+ _generic.get_usage = lambda response: _fake_extract_usage(response_payload=response)
144
+ _generic.calculate_cost = _fake_calculate_cost
145
+ _generic.bill_with_retry = _fake_bill_with_retry
146
+
118
147
  # Restore original sys.modules mappings to avoid side-effects for other tests
119
148
  for name, orig in _orig_sys_modules.items():
120
149
  if orig is None:
@@ -186,8 +186,8 @@ sys.modules['ulid'] = ulid_mod
186
186
  # Load the module directly from the functions/ tree so the test's fake sys.modules entries are respected
187
187
  repo_root = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..', '..'))
188
188
  functions_root = os.path.join(repo_root, 'functions')
189
- module_path = os.path.join(functions_root, 'api_v0_01', '_get_payment_details.py')
190
- spec = importlib.util.spec_from_file_location('api_v0_01._get_payment_details', module_path)
189
+ module_path = os.path.join(functions_root, 'api_v1_00', '_get_payment_details.py')
190
+ spec = importlib.util.spec_from_file_location('api_v1_00._get_payment_details', module_path)
191
191
  _get_payment_details = importlib.util.module_from_spec(spec)
192
192
  spec.loader.exec_module(_get_payment_details)
193
193