mage-ai 0.9.74__py3-none-any.whl → 0.9.79__py3-none-any.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.
- mage_ai/ai/llm_pipeline_wizard.py +6 -4
- mage_ai/ai/openai_client.py +7 -5
- mage_ai/api/policies/PipelineSchedulePolicy.py +1 -0
- mage_ai/api/presenters/PipelineSchedulePresenter.py +11 -2
- mage_ai/api/resources/GitFileResource.py +8 -0
- mage_ai/api/resources/PipelineScheduleResource.py +20 -14
- mage_ai/api/resources/PipelineTriggerResource.py +3 -1
- mage_ai/api/resources/SessionResource.py +2 -2
- mage_ai/api/resources/SyncResource.py +1 -1
- mage_ai/api/resources/UserResource.py +1 -1
- mage_ai/cli/main.py +8 -1
- mage_ai/data_cleaner/analysis/charts.py +1 -1
- mage_ai/data_cleaner/cleaning_rules/reformat_values.py +1 -1
- mage_ai/data_integrations/destinations/constants.py +3 -0
- mage_ai/data_integrations/sources/constants.py +2 -0
- mage_ai/data_preparation/executors/block_executor.py +8 -3
- mage_ai/data_preparation/executors/pipeline_executor.py +35 -19
- mage_ai/data_preparation/git/utils.py +2 -2
- mage_ai/data_preparation/logging/logger_manager.py +31 -2
- mage_ai/data_preparation/models/block/__init__.py +33 -27
- mage_ai/data_preparation/models/block/dbt/dbt_adapter.py +20 -8
- mage_ai/data_preparation/models/block/dynamic/constants.py +0 -1
- mage_ai/data_preparation/models/block/dynamic/counter.py +1 -3
- mage_ai/data_preparation/models/block/outputs.py +7 -1
- mage_ai/data_preparation/models/block/r/__init__.py +16 -5
- mage_ai/data_preparation/models/block/sql/__init__.py +2 -0
- mage_ai/data_preparation/models/block/sql/mssql.py +8 -0
- mage_ai/data_preparation/models/block/sql/utils/shared.py +6 -2
- mage_ai/data_preparation/models/constants.py +4 -1
- mage_ai/data_preparation/models/pipeline.py +11 -2
- mage_ai/data_preparation/models/project/__init__.py +3 -1
- mage_ai/data_preparation/models/triggers/__init__.py +1 -1
- mage_ai/data_preparation/storage/local_storage.py +4 -1
- mage_ai/data_preparation/templates/constants.py +7 -0
- mage_ai/data_preparation/templates/data_exporters/streaming/elasticsearch.yaml +3 -0
- mage_ai/data_preparation/templates/data_loaders/airtable.py +28 -0
- mage_ai/data_preparation/templates/data_loaders/streaming/nats.yaml +6 -3
- mage_ai/data_preparation/templates/repo/io_config.yaml +2 -0
- mage_ai/io/airtable.py +104 -0
- mage_ai/io/base.py +30 -1
- mage_ai/io/bigquery.py +36 -0
- mage_ai/io/config.py +6 -0
- mage_ai/io/mssql.py +21 -9
- mage_ai/io/mysql.py +6 -1
- mage_ai/io/oracledb.py +2 -4
- mage_ai/io/postgres.py +41 -19
- mage_ai/io/qdrant.py +1 -1
- mage_ai/io/redshift.py +13 -0
- mage_ai/io/sql.py +1 -0
- mage_ai/io/utils.py +18 -0
- mage_ai/orchestration/db/__init__.py +23 -3
- mage_ai/orchestration/db/migrations/versions/39d36f1dab73_create_genericjob.py +47 -0
- mage_ai/orchestration/db/models/oauth.py +2 -1
- mage_ai/orchestration/db/models/schedules.py +108 -6
- mage_ai/orchestration/db/models/schedules_project_platform.py +1 -1
- mage_ai/orchestration/db/models/secrets.py +11 -1
- mage_ai/orchestration/job_manager.py +19 -0
- mage_ai/orchestration/metrics/pipeline_run.py +1 -1
- mage_ai/orchestration/notification/sender.py +2 -2
- mage_ai/orchestration/pipeline_scheduler_original.py +150 -6
- mage_ai/orchestration/pipeline_scheduler_project_platform.py +4 -5
- mage_ai/orchestration/queue/config.py +11 -1
- mage_ai/orchestration/queue/process_queue.py +4 -0
- mage_ai/orchestration/utils/distributed_lock.py +8 -1
- mage_ai/orchestration/utils/resources.py +56 -2
- mage_ai/sample_datasets/salary_survey.csv +52 -52
- mage_ai/server/api/base.py +41 -0
- mage_ai/server/api/constants.py +1 -0
- mage_ai/server/api/triggers.py +9 -0
- mage_ai/server/constants.py +1 -1
- mage_ai/server/frontend_dist/404.html +3 -3
- mage_ai/server/frontend_dist/_next/static/TUo4RceCdMufBTBTq8CAq/_buildManifest.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/{1187-839336d276186105.js → 1187-4560c3895e1d7099.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/{1598-0adca9dce3ba4c60.js → 1598-cbf3f5a6078fc3f5.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/2717-638a944d24d5abde.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/3548-36f746b1824004f2.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/{3763-39a5174f6a3924db.js → 3763-aabe2703076636b0.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/3782-3e2acb5ed45b582b.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/449-5e2253c6aba42557.js +1 -0
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/5627-d5e559859dd0e1e0.js → frontend_dist/_next/static/chunks/5627-10e76bafa5a26f5f.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/{5699-e49718dfc9eb2854.js → 5699-e99379e332bd0b41.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/{7966-163da2621b8c987c.js → 7966-a5a7db345ce81263.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/_app-b697b35dfc4e6e26.js +2 -0
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/compute-ed67fa8e81662e8b.js → frontend_dist/_next/static/chunks/pages/compute-9e2dea78024e3bb4.js} +1 -1
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/files-e0ecd7ced09a63b2.js → frontend_dist/_next/static/chunks/pages/files-e08c7fe76f968f9c.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/global-data-products/{[...slug]-c7a729477ecda50e.js → [...slug]-30c3807057a4e65b.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/{global-data-products-fd6ae6a358a60a0c.js → global-data-products-8dcb3b31af9e0e39.js} +1 -1
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/global-hooks/[...slug]-8e50243797a7fe59.js → frontend_dist/_next/static/chunks/pages/global-hooks/[...slug]-85a64b64d27214b6.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/{global-hooks-d0c003446332dc0d.js → global-hooks-4ff959d51b8a9502.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/manage/{files-a69ed8e9f814490c.js → files-d08a460641d0efaa.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/manage/{overview-1aad7093c6d39257.js → overview-aae747f487e08d51.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/manage/{pipeline-runs-528d30e0d13b0cc7.js → pipeline-runs-09a842d64a6ada62.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/manage/{settings-fb9201d9cf63031d.js → settings-2e98e57d9376a458.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/manage/users/{[user]-000f5a980a07da39.js → [user]-7be6e41ad66089bb.js} +1 -1
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/manage/users/new-e4e613f6e817a733.js → frontend_dist/_next/static/chunks/pages/manage/users/new-4c088833063bfa07.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/manage/users-5db54821a3059c69.js +1 -0
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/manage-34d718b8a4066c23.js → frontend_dist/_next/static/chunks/pages/manage-868fcd8cbeb265f0.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/{oauth-3bfd1b8d7f036726.js → oauth-6ceceb62191dfe8a.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/overview-f65416f6dbe30ad3.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/{pipeline-runs-5f8c100e648efa8a.js → pipeline-runs-2d0136b51b57de93.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/backfills/{[...slug]-688c652f3296bb9c.js → [...slug]-1ad5238742e25b4c.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/backfills-bd11e87d026bfbf9.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/{dashboard-1236e36d39b1637d.js → dashboard-0f4f47f721b0723f.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/edit-5ae8efe9e0530212.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/logs-fe91dfb0091f6bc6.js +1 -0
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/monitors/block-runs-0d68d4bf6290fefb.js → frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/monitors/block-runs-cf794b2d22a80f31.js} +1 -1
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/monitors/block-runtime-9254358d58f07714.js → frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/monitors/block-runtime-a964caef91bed9e1.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/{monitors-821001e690caebe2.js → monitors-80bebb4401eefe25.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/runs/[run]-2eae7cb017027682.js +1 -0
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/runs-2d4b2a0800a66b33.js → frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/runs-776b2e5b0b6ceba8.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/settings-03d9bca3bc5e6088.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/syncs-707ed8ca942ca802.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/triggers/{[...slug]-259143ed3cf59e31.js → [...slug]-8429f17d4146e1ec.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/triggers-193045d9836d8d80.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines-d25d07db166cbb04.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/platform/global-hooks/{[...slug]-5eeec927e4202b63.js → [...slug]-6834ae87bd668cb2.js} +1 -1
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/platform/global-hooks-fbe9ad995d46d837.js → frontend_dist/_next/static/chunks/pages/platform/global-hooks-b3f7309a23e592b2.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/account/{profile-fc659962d4015cb3.js → profile-f8b7374385e1f1bf.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/platform/preferences-8de68502a9afa299.js +1 -0
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/settings/platform/settings-a4f88c334414402b.js → frontend_dist/_next/static/chunks/pages/settings/platform/settings-50fb6a34f3913f1f.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/permissions/{[...slug]-4deb9579ef99a3c6.js → [...slug]-2e5c098c21ea32b7.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/{permissions-e0cda2f2bfce8d61.js → permissions-54e4b15b9585bfc4.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/preferences-040f83d75d0f6537.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/roles/{[...slug]-910257d16c604ebd.js → [...slug]-95088f43034e3c95.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/{roles-4f7a0756806cee34.js → roles-e9149e1fcf218f42.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/{sync-data-208d6f955204d704.js → sync-data-75b67ae4a00818ef.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/users/{[...slug]-c89dc67e5a1706a8.js → [...slug]-557dda05ca6c6124.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/users-fa61dc6c1370e6a5.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/{sign-in-054b33312d3193c3.js → sign-in-593c40985d63fcf7.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/templates/{[...slug]-b6ed6a5d818bfd20.js → [...slug]-252c4b6b818345d5.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/{templates-852357bc983af2ea.js → templates-ca528bc607753ab8.js} +1 -1
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/terminal-1f9c56d671bbc67d.js → frontend_dist/_next/static/chunks/pages/terminal-287362c1defcc96b.js} +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/test-2f83af8c9f1378fe.js +1 -0
- mage_ai/server/frontend_dist/_next/static/chunks/pages/triggers-d9de73fb799efed8.js +1 -0
- mage_ai/server/{frontend_dist_base_path_template/_next/static/chunks/pages/version-control-ae3469b992a341d6.js → frontend_dist/_next/static/chunks/pages/version-control-573f0225d7a703ed.js} +1 -1
- mage_ai/server/frontend_dist/block-layout.html +3 -3
- mage_ai/server/frontend_dist/compute.html +6 -6
- mage_ai/server/frontend_dist/files.html +6 -6
- mage_ai/server/frontend_dist/global-data-products/[...slug].html +6 -6
- mage_ai/server/frontend_dist/global-data-products.html +6 -6
- mage_ai/server/frontend_dist/global-hooks/[...slug].html +6 -6
- mage_ai/server/frontend_dist/global-hooks.html +6 -6
- mage_ai/server/frontend_dist/index.html +3 -3
- mage_ai/server/frontend_dist/manage/files.html +6 -6
- mage_ai/server/frontend_dist/manage/overview.html +6 -6
- mage_ai/server/frontend_dist/manage/pipeline-runs.html +6 -6
- mage_ai/server/frontend_dist/manage/settings.html +6 -6
- mage_ai/server/frontend_dist/manage/users/[user].html +6 -6
- mage_ai/server/frontend_dist/manage/users/new.html +6 -6
- mage_ai/server/frontend_dist/manage/users.html +6 -6
- mage_ai/server/frontend_dist/manage.html +6 -6
- mage_ai/server/frontend_dist/oauth.html +5 -5
- mage_ai/server/frontend_dist/overview.html +6 -6
- mage_ai/server/frontend_dist/pipeline-runs.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/backfills/[...slug].html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/backfills.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/dashboard.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/edit.html +3 -3
- mage_ai/server/frontend_dist/pipelines/[pipeline]/logs.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/monitors/block-runs.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/monitors/block-runtime.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/monitors.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/runs/[run].html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/runs.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/settings.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/syncs.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/triggers/[...slug].html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline]/triggers.html +6 -6
- mage_ai/server/frontend_dist/pipelines/[pipeline].html +3 -3
- mage_ai/server/frontend_dist/pipelines.html +6 -6
- mage_ai/server/frontend_dist/platform/global-hooks/[...slug].html +6 -6
- mage_ai/server/frontend_dist/platform/global-hooks.html +6 -6
- mage_ai/server/frontend_dist/settings/account/profile.html +6 -6
- mage_ai/server/frontend_dist/settings/platform/preferences.html +6 -6
- mage_ai/server/frontend_dist/settings/platform/settings.html +6 -6
- mage_ai/server/frontend_dist/settings/workspace/permissions/[...slug].html +6 -6
- mage_ai/server/frontend_dist/settings/workspace/permissions.html +6 -6
- mage_ai/server/frontend_dist/settings/workspace/preferences.html +6 -6
- mage_ai/server/frontend_dist/settings/workspace/roles/[...slug].html +6 -6
- mage_ai/server/frontend_dist/settings/workspace/roles.html +6 -6
- mage_ai/server/frontend_dist/settings/workspace/sync-data.html +6 -6
- mage_ai/server/frontend_dist/settings/workspace/users/[...slug].html +6 -6
- mage_ai/server/frontend_dist/settings/workspace/users.html +6 -6
- mage_ai/server/frontend_dist/settings.html +3 -3
- mage_ai/server/frontend_dist/sign-in.html +7 -7
- mage_ai/server/frontend_dist/templates/[...slug].html +6 -6
- mage_ai/server/frontend_dist/templates.html +6 -6
- mage_ai/server/frontend_dist/terminal.html +6 -6
- mage_ai/server/frontend_dist/test.html +3 -3
- mage_ai/server/frontend_dist/triggers.html +6 -6
- mage_ai/server/frontend_dist/v2/canvas.html +2 -2
- mage_ai/server/frontend_dist/v2.html +2 -2
- mage_ai/server/frontend_dist/version-control.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/404.html +3 -3
- mage_ai/server/frontend_dist_base_path_template/_next/static/2QL-FT4lFR0a9bDZ7lNd9/_buildManifest.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/{1187-839336d276186105.js → 1187-4560c3895e1d7099.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/{1598-0adca9dce3ba4c60.js → 1598-cbf3f5a6078fc3f5.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/2717-638a944d24d5abde.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/3548-36f746b1824004f2.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/{3763-39a5174f6a3924db.js → 3763-aabe2703076636b0.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/3782-3e2acb5ed45b582b.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/449-5e2253c6aba42557.js +1 -0
- mage_ai/server/{frontend_dist/_next/static/chunks/5627-d5e559859dd0e1e0.js → frontend_dist_base_path_template/_next/static/chunks/5627-10e76bafa5a26f5f.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/{5699-e49718dfc9eb2854.js → 5699-e99379e332bd0b41.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/{7966-163da2621b8c987c.js → 7966-a5a7db345ce81263.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/_app-f205accb03b9ff43.js +2 -0
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/compute-ed67fa8e81662e8b.js → frontend_dist_base_path_template/_next/static/chunks/pages/compute-9e2dea78024e3bb4.js} +1 -1
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/files-e0ecd7ced09a63b2.js → frontend_dist_base_path_template/_next/static/chunks/pages/files-e08c7fe76f968f9c.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/global-data-products/{[...slug]-c7a729477ecda50e.js → [...slug]-30c3807057a4e65b.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/{global-data-products-fd6ae6a358a60a0c.js → global-data-products-8dcb3b31af9e0e39.js} +1 -1
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/global-hooks/[...slug]-8e50243797a7fe59.js → frontend_dist_base_path_template/_next/static/chunks/pages/global-hooks/[...slug]-85a64b64d27214b6.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/{global-hooks-d0c003446332dc0d.js → global-hooks-4ff959d51b8a9502.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/manage/{files-a69ed8e9f814490c.js → files-d08a460641d0efaa.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/manage/{overview-1aad7093c6d39257.js → overview-aae747f487e08d51.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/manage/{pipeline-runs-528d30e0d13b0cc7.js → pipeline-runs-09a842d64a6ada62.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/manage/{settings-fb9201d9cf63031d.js → settings-2e98e57d9376a458.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/manage/users/{[user]-000f5a980a07da39.js → [user]-7be6e41ad66089bb.js} +1 -1
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/manage/users/new-e4e613f6e817a733.js → frontend_dist_base_path_template/_next/static/chunks/pages/manage/users/new-4c088833063bfa07.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/manage/users-5db54821a3059c69.js +1 -0
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/manage-34d718b8a4066c23.js → frontend_dist_base_path_template/_next/static/chunks/pages/manage-868fcd8cbeb265f0.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/{oauth-3bfd1b8d7f036726.js → oauth-6ceceb62191dfe8a.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/overview-f65416f6dbe30ad3.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/{pipeline-runs-5f8c100e648efa8a.js → pipeline-runs-2d0136b51b57de93.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/backfills/{[...slug]-688c652f3296bb9c.js → [...slug]-1ad5238742e25b4c.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/backfills-bd11e87d026bfbf9.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/{dashboard-1236e36d39b1637d.js → dashboard-0f4f47f721b0723f.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/edit-5ae8efe9e0530212.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/logs-fe91dfb0091f6bc6.js +1 -0
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/monitors/block-runs-0d68d4bf6290fefb.js → frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/monitors/block-runs-cf794b2d22a80f31.js} +1 -1
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/monitors/block-runtime-9254358d58f07714.js → frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/monitors/block-runtime-a964caef91bed9e1.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/{monitors-821001e690caebe2.js → monitors-80bebb4401eefe25.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/runs/[run]-2eae7cb017027682.js +1 -0
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/runs-2d4b2a0800a66b33.js → frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/runs-776b2e5b0b6ceba8.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/settings-03d9bca3bc5e6088.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/syncs-707ed8ca942ca802.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/triggers/{[...slug]-259143ed3cf59e31.js → [...slug]-8429f17d4146e1ec.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/triggers-193045d9836d8d80.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines-d25d07db166cbb04.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/platform/global-hooks/{[...slug]-5eeec927e4202b63.js → [...slug]-6834ae87bd668cb2.js} +1 -1
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/platform/global-hooks-fbe9ad995d46d837.js → frontend_dist_base_path_template/_next/static/chunks/pages/platform/global-hooks-b3f7309a23e592b2.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/account/{profile-fc659962d4015cb3.js → profile-f8b7374385e1f1bf.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/platform/preferences-8de68502a9afa299.js +1 -0
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/settings/platform/settings-a4f88c334414402b.js → frontend_dist_base_path_template/_next/static/chunks/pages/settings/platform/settings-50fb6a34f3913f1f.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/permissions/{[...slug]-4deb9579ef99a3c6.js → [...slug]-2e5c098c21ea32b7.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/{permissions-e0cda2f2bfce8d61.js → permissions-54e4b15b9585bfc4.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/preferences-040f83d75d0f6537.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/roles/{[...slug]-910257d16c604ebd.js → [...slug]-95088f43034e3c95.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/{roles-4f7a0756806cee34.js → roles-e9149e1fcf218f42.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/{sync-data-208d6f955204d704.js → sync-data-75b67ae4a00818ef.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/users/{[...slug]-c89dc67e5a1706a8.js → [...slug]-557dda05ca6c6124.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/users-fa61dc6c1370e6a5.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/{sign-in-054b33312d3193c3.js → sign-in-593c40985d63fcf7.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/templates/{[...slug]-b6ed6a5d818bfd20.js → [...slug]-252c4b6b818345d5.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/{templates-852357bc983af2ea.js → templates-ca528bc607753ab8.js} +1 -1
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/terminal-1f9c56d671bbc67d.js → frontend_dist_base_path_template/_next/static/chunks/pages/terminal-287362c1defcc96b.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/test-2f83af8c9f1378fe.js +1 -0
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/triggers-d9de73fb799efed8.js +1 -0
- mage_ai/server/{frontend_dist/_next/static/chunks/pages/version-control-ae3469b992a341d6.js → frontend_dist_base_path_template/_next/static/chunks/pages/version-control-573f0225d7a703ed.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/{webpack-12ad70eb5c31aa92.js → webpack-5f4be622608d9267.js} +1 -1
- mage_ai/server/frontend_dist_base_path_template/block-layout.html +3 -3
- mage_ai/server/frontend_dist_base_path_template/compute.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/files.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/global-data-products/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/global-data-products.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/global-hooks/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/global-hooks.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/index.html +3 -3
- mage_ai/server/frontend_dist_base_path_template/manage/files.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/manage/overview.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/manage/pipeline-runs.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/manage/settings.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/manage/users/[user].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/manage/users/new.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/manage/users.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/manage.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/oauth.html +5 -5
- mage_ai/server/frontend_dist_base_path_template/overview.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipeline-runs.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/backfills/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/backfills.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/dashboard.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/edit.html +3 -3
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/logs.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/monitors/block-runs.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/monitors/block-runtime.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/monitors.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/runs/[run].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/runs.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/settings.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/syncs.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/triggers/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline]/triggers.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/pipelines/[pipeline].html +3 -3
- mage_ai/server/frontend_dist_base_path_template/pipelines.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/platform/global-hooks/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/platform/global-hooks.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/account/profile.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/platform/preferences.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/platform/settings.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/workspace/permissions/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/workspace/permissions.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/workspace/preferences.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/workspace/roles/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/workspace/roles.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/workspace/sync-data.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/workspace/users/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings/workspace/users.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/settings.html +3 -3
- mage_ai/server/frontend_dist_base_path_template/sign-in.html +7 -7
- mage_ai/server/frontend_dist_base_path_template/templates/[...slug].html +6 -6
- mage_ai/server/frontend_dist_base_path_template/templates.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/terminal.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/test.html +3 -3
- mage_ai/server/frontend_dist_base_path_template/triggers.html +6 -6
- mage_ai/server/frontend_dist_base_path_template/v2/canvas.html +2 -2
- mage_ai/server/frontend_dist_base_path_template/v2.html +2 -2
- mage_ai/server/frontend_dist_base_path_template/version-control.html +6 -6
- mage_ai/server/scheduler_manager.py +2 -0
- mage_ai/server/server.py +13 -0
- mage_ai/server/terminal_server.py +3 -0
- mage_ai/server/utils/output_display.py +5 -3
- mage_ai/services/aws/events/events.py +2 -2
- mage_ai/services/gcp/cloud_run/cloud_run.py +1 -1
- mage_ai/services/teams/config.py +13 -2
- mage_ai/services/teams/teams.py +13 -11
- mage_ai/settings/server.py +12 -1
- mage_ai/shared/constants.py +3 -1
- mage_ai/shared/croniter.py +1398 -0
- mage_ai/shared/enum.py +2 -5
- mage_ai/shared/environments.py +27 -3
- mage_ai/streaming/sinks/elasticsearch.py +15 -5
- mage_ai/streaming/sinks/kafka.py +21 -3
- mage_ai/streaming/sources/kafka.py +64 -7
- mage_ai/streaming/sources/kafka_oauth.py +182 -0
- mage_ai/tests/api/endpoints/test_blocks.py +1 -101
- mage_ai/tests/api/endpoints/test_configuration_options.py +1 -48
- mage_ai/tests/api/endpoints/test_configuration_options_project_platform.py +68 -0
- mage_ai/tests/api/endpoints/test_custom_designs.py +1 -106
- mage_ai/tests/api/endpoints/test_custom_designs_project_platform.py +132 -0
- mage_ai/tests/api/endpoints/test_dbt_blocks.py +111 -0
- mage_ai/tests/api/endpoints/test_file_contents.py +0 -48
- mage_ai/tests/api/endpoints/test_file_contents_with_project_platform.py +66 -0
- mage_ai/tests/api/endpoints/test_pipelines.py +0 -134
- mage_ai/tests/api/endpoints/test_pipelines_with_project_platform.py +143 -0
- mage_ai/tests/data_preparation/executors/test_block_executor.py +3 -3
- mage_ai/tests/data_preparation/logging/test_logger_manager.py +24 -5
- mage_ai/tests/data_preparation/models/block/dynamic/test_counter.py +1 -3
- mage_ai/tests/data_preparation/models/block/hook/test_hook_block.py +3 -3
- mage_ai/tests/data_preparation/models/test_block.py +7 -0
- mage_ai/tests/data_preparation/models/test_pipeline.py +55 -0
- mage_ai/tests/data_preparation/models/test_variable.py +2 -0
- mage_ai/tests/data_preparation/test_repo_manager.py +0 -63
- mage_ai/tests/data_preparation/test_repo_manager_project_platform.py +67 -0
- mage_ai/tests/data_preparation/test_variable_manager.py +0 -51
- mage_ai/tests/data_preparation/test_variable_manager_project_platform.py +41 -0
- mage_ai/tests/io/create_table/test_postgresql.py +28 -0
- mage_ai/tests/orchestration/db/models/test_schedules.py +1 -1
- mage_ai/tests/orchestration/notification/test_config.py +3 -3
- mage_ai/tests/orchestration/notification/test_sender.py +5 -1
- mage_ai/tests/orchestration/utils/__init__.py +0 -0
- mage_ai/tests/orchestration/utils/test_resources.py +235 -0
- mage_ai/tests/shared/test_croniter.py +2541 -0
- mage_ai/tests/streaming/sinks/test_kafka.py +130 -0
- mage_ai/tests/streaming/sources/test_kafka.py +125 -3
- mage_ai/tests/streaming/sources/test_kafka_oauth.py +208 -0
- mage_ai/tests/streaming/sources/test_kafka_raw_value.py +105 -0
- mage_ai/usage_statistics/logger.py +99 -15
- mage_ai-0.9.79.dist-info/METADATA +358 -0
- {mage_ai-0.9.74.dist-info → mage_ai-0.9.79.dist-info}/RECORD +377 -359
- {mage_ai-0.9.74.dist-info → mage_ai-0.9.79.dist-info}/WHEEL +1 -1
- mage_ai/server/frontend_dist/_next/static/chunks/1557-1ad0c64c2d08e569.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/2717-d65056b6b5e124eb.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/3548-b2c5edfb710886a6.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/3782-4b3091e550f809a2.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/_app-5bdff745074fb350.js +0 -2
- mage_ai/server/frontend_dist/_next/static/chunks/pages/manage/users-a5e9d77ed5b50205.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/overview-69af3253ad0d0d89.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/backfills-fe112809feb25b05.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/edit-e9ca358209cdf93d.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/logs-a29f1615d2e7d330.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/runs/[run]-6d382ae5bad9745c.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/settings-cd49372ae1702963.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/syncs-135be8974f7f5f2b.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines/[pipeline]/triggers-3af13e89adff4d6c.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/pipelines-b578b075a8d857e3.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/platform/preferences-058d283ee178c038.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/preferences-7b02bb70462144cb.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/settings/workspace/users-5212c01a9dc558da.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/test-86b12cc12d4a625f.js +0 -1
- mage_ai/server/frontend_dist/_next/static/chunks/pages/triggers-2481c40b18d5b6d4.js +0 -1
- mage_ai/server/frontend_dist/_next/static/pLWT6Sqd09xYpufCVIqnz/_buildManifest.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/JQewSAObpbhO0wrdAM6Ng/_buildManifest.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/1557-1ad0c64c2d08e569.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/2717-d65056b6b5e124eb.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/3548-b2c5edfb710886a6.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/3782-4b3091e550f809a2.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/_app-90de19bc03f1484b.js +0 -2
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/manage/users-a5e9d77ed5b50205.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/overview-69af3253ad0d0d89.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/backfills-fe112809feb25b05.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/edit-e9ca358209cdf93d.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/logs-a29f1615d2e7d330.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/runs/[run]-6d382ae5bad9745c.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/settings-cd49372ae1702963.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/syncs-135be8974f7f5f2b.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines/[pipeline]/triggers-3af13e89adff4d6c.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/pipelines-b578b075a8d857e3.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/platform/preferences-058d283ee178c038.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/preferences-7b02bb70462144cb.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/settings/workspace/users-5212c01a9dc558da.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/test-86b12cc12d4a625f.js +0 -1
- mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/triggers-2481c40b18d5b6d4.js +0 -1
- mage_ai-0.9.74.dist-info/METADATA +0 -544
- /mage_ai/server/frontend_dist/_next/static/{pLWT6Sqd09xYpufCVIqnz → TUo4RceCdMufBTBTq8CAq}/_ssgManifest.js +0 -0
- /mage_ai/server/frontend_dist/_next/static/chunks/pages/{_app-5bdff745074fb350.js.LICENSE.txt → _app-b697b35dfc4e6e26.js.LICENSE.txt} +0 -0
- /mage_ai/server/frontend_dist_base_path_template/_next/static/{JQewSAObpbhO0wrdAM6Ng → 2QL-FT4lFR0a9bDZ7lNd9}/_ssgManifest.js +0 -0
- /mage_ai/server/frontend_dist_base_path_template/_next/static/chunks/pages/{_app-90de19bc03f1484b.js.LICENSE.txt → _app-f205accb03b9ff43.js.LICENSE.txt} +0 -0
- {mage_ai-0.9.74.dist-info → mage_ai-0.9.79.dist-info}/entry_points.txt +0 -0
- {mage_ai-0.9.74.dist-info → mage_ai-0.9.79.dist-info/licenses}/LICENSE +0 -0
- {mage_ai-0.9.74.dist-info → mage_ai-0.9.79.dist-info}/top_level.txt +0 -0
|
@@ -1,5 +1,3 @@
|
|
|
1
|
-
from unittest.mock import patch
|
|
2
|
-
|
|
3
1
|
import pandas as pd
|
|
4
2
|
from pandas.testing import assert_frame_equal
|
|
5
3
|
|
|
@@ -10,12 +8,10 @@ from mage_ai.data_preparation.repo_manager import get_repo_config
|
|
|
10
8
|
from mage_ai.data_preparation.variable_manager import (
|
|
11
9
|
VariableManager,
|
|
12
10
|
get_global_variable,
|
|
13
|
-
get_global_variables,
|
|
14
11
|
set_global_variable,
|
|
15
12
|
)
|
|
16
13
|
from mage_ai.settings.repo import set_repo_path
|
|
17
14
|
from mage_ai.tests.base_test import DBTestCase
|
|
18
|
-
from mage_ai.tests.shared.mixins import ProjectPlatformMixin
|
|
19
15
|
|
|
20
16
|
|
|
21
17
|
class VariableManagerTest(DBTestCase):
|
|
@@ -128,50 +124,3 @@ class VariableManagerTest(DBTestCase):
|
|
|
128
124
|
# get_global_variables(None, pipeline=pipeline), pipeline.variables
|
|
129
125
|
# )
|
|
130
126
|
# self.assertEqual(get_global_variables(pipeline.uuid), pipeline.variables)
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
class VariableManagerProjectPlatformTests(ProjectPlatformMixin):
|
|
134
|
-
def test_get_global_variable(self):
|
|
135
|
-
with patch(
|
|
136
|
-
"mage_ai.data_preparation.variable_manager.project_platform_activated",
|
|
137
|
-
lambda: True,
|
|
138
|
-
):
|
|
139
|
-
with patch(
|
|
140
|
-
"mage_ai.data_preparation.models.pipeline.project_platform_activated",
|
|
141
|
-
lambda: True,
|
|
142
|
-
):
|
|
143
|
-
for settings in self.repo_paths.values():
|
|
144
|
-
pipeline = Pipeline.create(
|
|
145
|
-
self.faker.unique.name(),
|
|
146
|
-
repo_path=settings["full_path"],
|
|
147
|
-
)
|
|
148
|
-
value = self.faker.unique.name()
|
|
149
|
-
pipeline.variables = dict(mage=value)
|
|
150
|
-
pipeline.save()
|
|
151
|
-
|
|
152
|
-
self.assertEqual(get_global_variable(pipeline.uuid, "mage"), value)
|
|
153
|
-
|
|
154
|
-
def test_get_global_variables(self):
|
|
155
|
-
with patch(
|
|
156
|
-
"mage_ai.data_preparation.variable_manager.project_platform_activated",
|
|
157
|
-
lambda: True,
|
|
158
|
-
):
|
|
159
|
-
with patch(
|
|
160
|
-
"mage_ai.data_preparation.models.pipeline.project_platform_activated",
|
|
161
|
-
lambda: True,
|
|
162
|
-
):
|
|
163
|
-
for settings in self.repo_paths.values():
|
|
164
|
-
pipeline = Pipeline.create(
|
|
165
|
-
self.faker.unique.name(),
|
|
166
|
-
repo_path=settings["full_path"],
|
|
167
|
-
)
|
|
168
|
-
pipeline.variables = self.faker.unique.name()
|
|
169
|
-
pipeline.save()
|
|
170
|
-
|
|
171
|
-
self.assertEqual(
|
|
172
|
-
get_global_variables(None, pipeline=pipeline),
|
|
173
|
-
pipeline.variables,
|
|
174
|
-
)
|
|
175
|
-
self.assertEqual(
|
|
176
|
-
get_global_variables(pipeline.uuid), pipeline.variables
|
|
177
|
-
)
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
from unittest.mock import patch
|
|
2
|
+
|
|
3
|
+
from mage_ai.data_preparation.models.pipeline import Pipeline
|
|
4
|
+
from mage_ai.data_preparation.variable_manager import (
|
|
5
|
+
get_global_variable,
|
|
6
|
+
get_global_variables,
|
|
7
|
+
)
|
|
8
|
+
from mage_ai.tests.shared.mixins import ProjectPlatformMixin
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
@patch('mage_ai.data_preparation.models.pipeline.project_platform_activated', lambda: True)
|
|
12
|
+
@patch('mage_ai.data_preparation.variable_manager.project_platform_activated', lambda: True)
|
|
13
|
+
class VariableManagerProjectPlatformTests(ProjectPlatformMixin):
|
|
14
|
+
def test_get_global_variable(self):
|
|
15
|
+
for settings in self.repo_paths.values():
|
|
16
|
+
pipeline = Pipeline.create(
|
|
17
|
+
self.faker.unique.name(),
|
|
18
|
+
repo_path=settings["full_path"],
|
|
19
|
+
)
|
|
20
|
+
value = self.faker.unique.name()
|
|
21
|
+
pipeline.variables = dict(mage=value)
|
|
22
|
+
pipeline.save()
|
|
23
|
+
|
|
24
|
+
self.assertEqual(get_global_variable(pipeline.uuid, "mage"), value)
|
|
25
|
+
|
|
26
|
+
def test_get_global_variables(self):
|
|
27
|
+
for settings in self.repo_paths.values():
|
|
28
|
+
pipeline = Pipeline.create(
|
|
29
|
+
self.faker.unique.name(),
|
|
30
|
+
repo_path=settings["full_path"],
|
|
31
|
+
)
|
|
32
|
+
pipeline.variables = self.faker.unique.name()
|
|
33
|
+
pipeline.save()
|
|
34
|
+
|
|
35
|
+
self.assertEqual(
|
|
36
|
+
get_global_variables(None, pipeline=pipeline),
|
|
37
|
+
pipeline.variables,
|
|
38
|
+
)
|
|
39
|
+
self.assertEqual(
|
|
40
|
+
get_global_variables(pipeline.uuid), pipeline.variables
|
|
41
|
+
)
|
|
@@ -1,7 +1,9 @@
|
|
|
1
1
|
import pandas as pd
|
|
2
|
+
import simplejson
|
|
2
3
|
|
|
3
4
|
from mage_ai.io.export_utils import infer_dtypes
|
|
4
5
|
from mage_ai.io.postgres import Postgres
|
|
6
|
+
from mage_ai.shared.parsers import encode_complex
|
|
5
7
|
from mage_ai.tests.base_test import DBTestCase
|
|
6
8
|
|
|
7
9
|
|
|
@@ -26,3 +28,29 @@ class TestTablePostgres(DBTestCase):
|
|
|
26
28
|
)
|
|
27
29
|
self.assertEqual('CREATE TABLE Test.Test ("varchar_time" text,"datetime_time" TIMESTAMP);',
|
|
28
30
|
query)
|
|
31
|
+
|
|
32
|
+
def test_clean_array_value(self):
|
|
33
|
+
test_cases = [
|
|
34
|
+
[],
|
|
35
|
+
[123],
|
|
36
|
+
[['àabc', 'deèéf']],
|
|
37
|
+
[['08:00', '12:00'], ['15:00', '20:00']],
|
|
38
|
+
[['08:00', '12:00'], []],
|
|
39
|
+
]
|
|
40
|
+
expected = [
|
|
41
|
+
'{}',
|
|
42
|
+
'{123}',
|
|
43
|
+
'{{"àabc", "deèéf"}}',
|
|
44
|
+
'{{"08:00", "12:00"}, {"15:00", "20:00"}}',
|
|
45
|
+
'{{"08:00", "12:00"}, {}}',
|
|
46
|
+
]
|
|
47
|
+
for val, expected_val in zip(test_cases, expected):
|
|
48
|
+
cleaned_array_value = Postgres._clean_array_value(
|
|
49
|
+
simplejson.dumps(
|
|
50
|
+
val,
|
|
51
|
+
default=encode_complex,
|
|
52
|
+
ensure_ascii=False,
|
|
53
|
+
ignore_nan=True,
|
|
54
|
+
)
|
|
55
|
+
)
|
|
56
|
+
self.assertEqual(cleaned_array_value, expected_val)
|
|
@@ -3,7 +3,6 @@ from datetime import datetime, timedelta, timezone
|
|
|
3
3
|
from unittest.mock import Mock, patch
|
|
4
4
|
from uuid import uuid4
|
|
5
5
|
|
|
6
|
-
from croniter import croniter
|
|
7
6
|
from freezegun import freeze_time
|
|
8
7
|
|
|
9
8
|
from mage_ai.data_preparation.models.block import Block
|
|
@@ -22,6 +21,7 @@ from mage_ai.orchestration.db.models.schedules import (
|
|
|
22
21
|
PipelineSchedule,
|
|
23
22
|
)
|
|
24
23
|
from mage_ai.orchestration.pipeline_scheduler import configure_pipeline_run_payload
|
|
24
|
+
from mage_ai.shared.croniter import croniter
|
|
25
25
|
|
|
26
26
|
# from mage_ai.settings.utils import base_repo_path
|
|
27
27
|
from mage_ai.shared.hash import merge_dict
|
|
@@ -2,10 +2,10 @@ from mage_ai.orchestration.notification.config import NotificationConfig
|
|
|
2
2
|
from mage_ai.tests.base_test import TestCase
|
|
3
3
|
from mage_ai.tests.orchestration.notification.constants import (
|
|
4
4
|
EMAIL_NOTIFICATION_CONFIG,
|
|
5
|
+
GOOGLE_CHAT_NOTIFICATION_CONFIG,
|
|
6
|
+
OPSGENIE_NOTIFICATION_CONFIG,
|
|
5
7
|
SLACK_NOTIFICATION_CONFIG,
|
|
6
8
|
TEAMS_NOTIFICATION_CONFIG,
|
|
7
|
-
GOOGLE_CHAT_NOTIFICATION_CONFIG,
|
|
8
|
-
OPSGENIE_NOTIFICATION_CONFIG
|
|
9
9
|
)
|
|
10
10
|
|
|
11
11
|
|
|
@@ -43,7 +43,7 @@ class NotificationConfigTests(TestCase):
|
|
|
43
43
|
self.assertIsNone(config3.email_config)
|
|
44
44
|
self.assertEqual(config3.slack_config.webhook_url, 'test_webhook_url')
|
|
45
45
|
|
|
46
|
-
self.assertEqual(config4.teams_config.webhook_url, 'test_webhook_url')
|
|
46
|
+
self.assertEqual(config4.teams_config.webhook_url, ['test_webhook_url'])
|
|
47
47
|
|
|
48
48
|
self.assertEqual(config5.google_chat_config.webhook_url, 'test_webhook_url')
|
|
49
49
|
|
|
@@ -112,9 +112,11 @@ class NotificationSenderTests(DBTestCase):
|
|
|
112
112
|
f'`{pipeline_run.pipeline_schedule.name}` '
|
|
113
113
|
f'at execution time `{pipeline_run.execution_date}`. Error: None'
|
|
114
114
|
)
|
|
115
|
+
title = 'Failed to run Mage pipeline test_pipeline'
|
|
115
116
|
mock_send_teams_message.assert_called_once_with(
|
|
116
117
|
notification_config.teams_config,
|
|
117
118
|
message,
|
|
119
|
+
title
|
|
118
120
|
)
|
|
119
121
|
|
|
120
122
|
@patch('mage_ai.orchestration.notification.sender.send_teams_message')
|
|
@@ -131,9 +133,11 @@ class NotificationSenderTests(DBTestCase):
|
|
|
131
133
|
f'`{pipeline_run.pipeline_schedule.name}` '
|
|
132
134
|
f'at execution time `{pipeline_run.execution_date}`.'
|
|
133
135
|
)
|
|
136
|
+
title = 'Successfully ran Pipeline test_pipeline'
|
|
134
137
|
mock_send_teams_message.assert_called_once_with(
|
|
135
138
|
notification_config.teams_config,
|
|
136
139
|
message,
|
|
140
|
+
title
|
|
137
141
|
)
|
|
138
142
|
|
|
139
143
|
@patch('mage_ai.orchestration.notification.sender.send_teams_message')
|
|
@@ -162,4 +166,4 @@ class NotificationSenderTests(DBTestCase):
|
|
|
162
166
|
notification_config.opsgenie_config,
|
|
163
167
|
message=ANY,
|
|
164
168
|
description=ANY,
|
|
165
|
-
)
|
|
169
|
+
)
|
|
File without changes
|
|
@@ -0,0 +1,235 @@
|
|
|
1
|
+
import os
|
|
2
|
+
import tempfile
|
|
3
|
+
import unittest
|
|
4
|
+
from unittest.mock import MagicMock, patch
|
|
5
|
+
|
|
6
|
+
from mage_ai.orchestration.utils.resources import get_memory
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
class TestGetMemory(unittest.TestCase):
|
|
10
|
+
"""Test cases for the get_memory function."""
|
|
11
|
+
|
|
12
|
+
@patch('os.path.exists')
|
|
13
|
+
@patch('builtins.open', create=True)
|
|
14
|
+
def test_get_memory_cgroup_v1(self, mock_open, mock_exists):
|
|
15
|
+
"""Test memory reading from cgroup v1 files."""
|
|
16
|
+
# Setup: cgroup v1 paths exist
|
|
17
|
+
def exists_side_effect(path):
|
|
18
|
+
return path in [
|
|
19
|
+
"/sys/fs/cgroup/memory/memory.usage_in_bytes",
|
|
20
|
+
"/sys/fs/cgroup/memory/memory.limit_in_bytes"
|
|
21
|
+
]
|
|
22
|
+
|
|
23
|
+
mock_exists.side_effect = exists_side_effect
|
|
24
|
+
|
|
25
|
+
# Mock file reads: 1387 MB usage, 4768 MB limit (from the issue)
|
|
26
|
+
usage_bytes = 1387 * 1024 * 1024
|
|
27
|
+
limit_bytes = 4768 * 1024 * 1024
|
|
28
|
+
|
|
29
|
+
mock_file_handles = {
|
|
30
|
+
"/sys/fs/cgroup/memory/memory.usage_in_bytes": MagicMock(
|
|
31
|
+
__enter__=lambda self: self,
|
|
32
|
+
__exit__=lambda self, *args: None,
|
|
33
|
+
read=lambda: str(usage_bytes)
|
|
34
|
+
),
|
|
35
|
+
"/sys/fs/cgroup/memory/memory.limit_in_bytes": MagicMock(
|
|
36
|
+
__enter__=lambda self: self,
|
|
37
|
+
__exit__=lambda self, *args: None,
|
|
38
|
+
read=lambda: str(limit_bytes)
|
|
39
|
+
),
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
def open_side_effect(path, mode='r'):
|
|
43
|
+
return mock_file_handles.get(path)
|
|
44
|
+
|
|
45
|
+
mock_open.side_effect = open_side_effect
|
|
46
|
+
|
|
47
|
+
# Execute
|
|
48
|
+
free_memory, used_memory, total_memory = get_memory()
|
|
49
|
+
|
|
50
|
+
# Assert
|
|
51
|
+
self.assertAlmostEqual(used_memory, 1387.0, places=1)
|
|
52
|
+
self.assertAlmostEqual(total_memory, 4768.0, places=1)
|
|
53
|
+
self.assertAlmostEqual(free_memory, 3381.0, places=1)
|
|
54
|
+
|
|
55
|
+
@patch('os.path.exists')
|
|
56
|
+
@patch('builtins.open', create=True)
|
|
57
|
+
def test_get_memory_cgroup_v2(self, mock_open, mock_exists):
|
|
58
|
+
"""Test memory reading from cgroup v2 files."""
|
|
59
|
+
# Setup: cgroup v2 paths exist, v1 don't
|
|
60
|
+
def exists_side_effect(path):
|
|
61
|
+
return path in [
|
|
62
|
+
"/sys/fs/cgroup/memory.current",
|
|
63
|
+
"/sys/fs/cgroup/memory.max"
|
|
64
|
+
]
|
|
65
|
+
|
|
66
|
+
mock_exists.side_effect = exists_side_effect
|
|
67
|
+
|
|
68
|
+
# Mock file reads
|
|
69
|
+
usage_bytes = 1500 * 1024 * 1024
|
|
70
|
+
limit_bytes = 5000 * 1024 * 1024
|
|
71
|
+
|
|
72
|
+
# Create mock read() return value with strip() method
|
|
73
|
+
usage_read_value = MagicMock()
|
|
74
|
+
usage_read_value.strip.return_value = str(usage_bytes)
|
|
75
|
+
|
|
76
|
+
limit_read_value = MagicMock()
|
|
77
|
+
limit_read_value.strip.return_value = str(limit_bytes)
|
|
78
|
+
|
|
79
|
+
mock_file_handles = {
|
|
80
|
+
"/sys/fs/cgroup/memory.current": MagicMock(
|
|
81
|
+
__enter__=lambda self: self,
|
|
82
|
+
__exit__=lambda self, *args: None,
|
|
83
|
+
read=lambda: usage_read_value
|
|
84
|
+
),
|
|
85
|
+
"/sys/fs/cgroup/memory.max": MagicMock(
|
|
86
|
+
__enter__=lambda self: self,
|
|
87
|
+
__exit__=lambda self, *args: None,
|
|
88
|
+
read=lambda: limit_read_value
|
|
89
|
+
),
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
def open_side_effect(path, mode='r'):
|
|
93
|
+
return mock_file_handles.get(path)
|
|
94
|
+
|
|
95
|
+
mock_open.side_effect = open_side_effect
|
|
96
|
+
|
|
97
|
+
# Execute
|
|
98
|
+
free_memory, used_memory, total_memory = get_memory()
|
|
99
|
+
|
|
100
|
+
# Assert
|
|
101
|
+
self.assertAlmostEqual(used_memory, 1500.0, places=1)
|
|
102
|
+
self.assertAlmostEqual(total_memory, 5000.0, places=1)
|
|
103
|
+
self.assertAlmostEqual(free_memory, 3500.0, places=1)
|
|
104
|
+
|
|
105
|
+
@patch('os.path.exists')
|
|
106
|
+
@patch('builtins.open', create=True)
|
|
107
|
+
def test_get_memory_cgroup_v2_max_limit(self, mock_open, mock_exists):
|
|
108
|
+
"""Test memory reading from cgroup v2 with 'max' limit (no limit)."""
|
|
109
|
+
# Setup: cgroup v2 with max limit
|
|
110
|
+
def exists_side_effect(path):
|
|
111
|
+
return path in [
|
|
112
|
+
"/sys/fs/cgroup/memory.current",
|
|
113
|
+
"/sys/fs/cgroup/memory.max"
|
|
114
|
+
]
|
|
115
|
+
|
|
116
|
+
mock_exists.side_effect = exists_side_effect
|
|
117
|
+
|
|
118
|
+
usage_bytes = 1500 * 1024 * 1024
|
|
119
|
+
|
|
120
|
+
# Create mock read() return value with strip() method
|
|
121
|
+
usage_read_value = MagicMock()
|
|
122
|
+
usage_read_value.strip.return_value = str(usage_bytes)
|
|
123
|
+
|
|
124
|
+
limit_read_value = MagicMock()
|
|
125
|
+
limit_read_value.strip.return_value = "max"
|
|
126
|
+
|
|
127
|
+
mock_file_handles = {
|
|
128
|
+
"/sys/fs/cgroup/memory.current": MagicMock(
|
|
129
|
+
__enter__=lambda self: self,
|
|
130
|
+
__exit__=lambda self, *args: None,
|
|
131
|
+
read=lambda: usage_read_value
|
|
132
|
+
),
|
|
133
|
+
"/sys/fs/cgroup/memory.max": MagicMock(
|
|
134
|
+
__enter__=lambda self: self,
|
|
135
|
+
__exit__=lambda self, *args: None,
|
|
136
|
+
read=lambda: limit_read_value
|
|
137
|
+
),
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
def open_side_effect(path, mode='r'):
|
|
141
|
+
return mock_file_handles.get(path)
|
|
142
|
+
|
|
143
|
+
mock_open.side_effect = open_side_effect
|
|
144
|
+
|
|
145
|
+
# With 'max' limit, should fall back to free command
|
|
146
|
+
# We'll mock subprocess as well
|
|
147
|
+
with patch('subprocess.check_output') as mock_subprocess:
|
|
148
|
+
mock_subprocess.return_value = b"Mem: 8000 4000 4000\nSwap: 0 0 0\nTotal: 8000 4000 4000"
|
|
149
|
+
|
|
150
|
+
free_memory, used_memory, total_memory = get_memory()
|
|
151
|
+
|
|
152
|
+
# Should use free command fallback
|
|
153
|
+
self.assertEqual(total_memory, 8000.0)
|
|
154
|
+
self.assertEqual(used_memory, 4000.0)
|
|
155
|
+
self.assertEqual(free_memory, 4000.0)
|
|
156
|
+
|
|
157
|
+
@patch('os.path.exists')
|
|
158
|
+
@patch('subprocess.check_output')
|
|
159
|
+
def test_get_memory_fallback_to_free(self, mock_subprocess, mock_exists):
|
|
160
|
+
"""Test fallback to 'free' command when cgroup files don't exist."""
|
|
161
|
+
# Setup: no cgroup files exist
|
|
162
|
+
mock_exists.return_value = False
|
|
163
|
+
|
|
164
|
+
# Mock free command output
|
|
165
|
+
mock_subprocess.return_value = b"Mem: 4570 1921 2533\nSwap: 0 0 0\nTotal: 4570 1921 2533"
|
|
166
|
+
|
|
167
|
+
# Execute
|
|
168
|
+
free_memory, used_memory, total_memory = get_memory()
|
|
169
|
+
|
|
170
|
+
# Assert - values from the issue's screenshot
|
|
171
|
+
self.assertEqual(total_memory, 4570.0)
|
|
172
|
+
self.assertEqual(used_memory, 1921.0)
|
|
173
|
+
self.assertEqual(free_memory, 2533.0)
|
|
174
|
+
|
|
175
|
+
@patch('os.name', 'nt')
|
|
176
|
+
def test_get_memory_windows(self):
|
|
177
|
+
"""Test that Windows returns None values."""
|
|
178
|
+
free_memory, used_memory, total_memory = get_memory()
|
|
179
|
+
|
|
180
|
+
self.assertIsNone(free_memory)
|
|
181
|
+
self.assertIsNone(used_memory)
|
|
182
|
+
self.assertIsNone(total_memory)
|
|
183
|
+
|
|
184
|
+
@patch('os.path.exists')
|
|
185
|
+
@patch('builtins.open', create=True)
|
|
186
|
+
def test_get_memory_cgroup_v1_no_limit(self, mock_open, mock_exists):
|
|
187
|
+
"""Test memory reading with very large limit value (no limit set)."""
|
|
188
|
+
# Setup: cgroup v1 paths exist
|
|
189
|
+
def exists_side_effect(path):
|
|
190
|
+
return path in [
|
|
191
|
+
"/sys/fs/cgroup/memory/memory.usage_in_bytes",
|
|
192
|
+
"/sys/fs/cgroup/memory/memory.limit_in_bytes"
|
|
193
|
+
]
|
|
194
|
+
|
|
195
|
+
mock_exists.side_effect = exists_side_effect
|
|
196
|
+
|
|
197
|
+
usage_bytes = 1500 * 1024 * 1024
|
|
198
|
+
# Very large limit indicates no limit
|
|
199
|
+
limit_bytes = 9223372036854771712
|
|
200
|
+
|
|
201
|
+
# Create mock read() return value with strip() method
|
|
202
|
+
usage_read_value = MagicMock()
|
|
203
|
+
usage_read_value.strip.return_value = str(usage_bytes)
|
|
204
|
+
|
|
205
|
+
limit_read_value = MagicMock()
|
|
206
|
+
limit_read_value.strip.return_value = str(limit_bytes)
|
|
207
|
+
|
|
208
|
+
mock_file_handles = {
|
|
209
|
+
"/sys/fs/cgroup/memory/memory.usage_in_bytes": MagicMock(
|
|
210
|
+
__enter__=lambda self: self,
|
|
211
|
+
__exit__=lambda self, *args: None,
|
|
212
|
+
read=lambda: usage_read_value
|
|
213
|
+
),
|
|
214
|
+
"/sys/fs/cgroup/memory/memory.limit_in_bytes": MagicMock(
|
|
215
|
+
__enter__=lambda self: self,
|
|
216
|
+
__exit__=lambda self, *args: None,
|
|
217
|
+
read=lambda: limit_read_value
|
|
218
|
+
),
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
def open_side_effect(path, mode='r'):
|
|
222
|
+
return mock_file_handles.get(path)
|
|
223
|
+
|
|
224
|
+
mock_open.side_effect = open_side_effect
|
|
225
|
+
|
|
226
|
+
# With no real limit, should fall back to free command
|
|
227
|
+
with patch('subprocess.check_output') as mock_subprocess:
|
|
228
|
+
mock_subprocess.return_value = b"Mem: 8000 4000 4000\nSwap: 0 0 0\nTotal: 8000 4000 4000"
|
|
229
|
+
|
|
230
|
+
free_memory, used_memory, total_memory = get_memory()
|
|
231
|
+
|
|
232
|
+
# Should use free command fallback
|
|
233
|
+
self.assertEqual(total_memory, 8000.0)
|
|
234
|
+
self.assertEqual(used_memory, 4000.0)
|
|
235
|
+
self.assertEqual(free_memory, 4000.0)
|