uffizzi_core 0.1.3
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/LICENSE +201 -0
- data/README.md +46 -0
- data/Rakefile +31 -0
- data/app/assets/config/uffizzi_core_manifest.js +1 -0
- data/app/assets/stylesheets/uffizzi_core/application.css +15 -0
- data/app/clients/uffizzi_core/amazon_registry_client.rb +18 -0
- data/app/clients/uffizzi_core/azure_registry_client/request_result.rb +5 -0
- data/app/clients/uffizzi_core/azure_registry_client.rb +42 -0
- data/app/clients/uffizzi_core/controller_client/request_result.rb +5 -0
- data/app/clients/uffizzi_core/controller_client.rb +106 -0
- data/app/clients/uffizzi_core/docker_hub_client/request_result.rb +7 -0
- data/app/clients/uffizzi_core/docker_hub_client.rb +139 -0
- data/app/clients/uffizzi_core/github/app_client.rb +19 -0
- data/app/clients/uffizzi_core/github/installation_client.rb +11 -0
- data/app/clients/uffizzi_core/github/user_client.rb +51 -0
- data/app/clients/uffizzi_core/google_registry_client/request_result.rb +5 -0
- data/app/clients/uffizzi_core/google_registry_client.rb +42 -0
- data/app/contexts/uffizzi_core/base_context.rb +12 -0
- data/app/contexts/uffizzi_core/project_context.rb +13 -0
- data/app/contexts/uffizzi_core/webhooks_context.rb +9 -0
- data/app/controllers/concerns/uffizzi_core/auth_management.rb +23 -0
- data/app/controllers/concerns/uffizzi_core/authorization_concern.rb +38 -0
- data/app/controllers/concerns/uffizzi_core/dependency_injection_concern.rb +19 -0
- data/app/controllers/uffizzi_core/api/cli/v1/account/application_controller.rb +7 -0
- data/app/controllers/uffizzi_core/api/cli/v1/account/credentials_controller.rb +55 -0
- data/app/controllers/uffizzi_core/api/cli/v1/application_controller.rb +5 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/application_controller.rb +11 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/compose_files_controller.rb +93 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/deployments/activity_items_controller.rb +36 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/deployments/application_controller.rb +7 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/deployments/containers/application_controller.rb +8 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/deployments/containers/logs_controller.rb +27 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/deployments/containers_controller.rb +24 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/deployments/events_controller.rb +29 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/deployments_controller.rb +148 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects/secrets_controller.rb +69 -0
- data/app/controllers/uffizzi_core/api/cli/v1/projects_controller.rb +19 -0
- data/app/controllers/uffizzi_core/api/cli/v1/sessions_controller.rb +43 -0
- data/app/controllers/uffizzi_core/application_controller.rb +51 -0
- data/app/errors/uffizzi_core/compose_file/build_error.rb +4 -0
- data/app/errors/uffizzi_core/compose_file/credential_error.rb +4 -0
- data/app/errors/uffizzi_core/compose_file/not_found_error.rb +4 -0
- data/app/errors/uffizzi_core/compose_file/parse_error.rb +4 -0
- data/app/errors/uffizzi_core/compose_file/secrets_error.rb +4 -0
- data/app/errors/uffizzi_core/deployment_not_found_error.rb +10 -0
- data/app/forms/uffizzi_core/api/cli/v1/account/credential/create_form.rb +26 -0
- data/app/forms/uffizzi_core/api/cli/v1/compose_file/check_credentials_form.rb +21 -0
- data/app/forms/uffizzi_core/api/cli/v1/compose_file/cli_form.rb +39 -0
- data/app/forms/uffizzi_core/api/cli/v1/compose_file/create_form.rb +13 -0
- data/app/forms/uffizzi_core/api/cli/v1/compose_file/template_form.rb +44 -0
- data/app/forms/uffizzi_core/api/cli/v1/compose_file/update_form.rb +9 -0
- data/app/forms/uffizzi_core/api/cli/v1/config_file/create_form.rb +11 -0
- data/app/forms/uffizzi_core/api/cli/v1/deployment/create_form.rb +91 -0
- data/app/forms/uffizzi_core/api/cli/v1/project/delete_secret_form.rb +27 -0
- data/app/forms/uffizzi_core/api/cli/v1/project/update_form.rb +40 -0
- data/app/forms/uffizzi_core/api/cli/v1/session_create_form.rb +29 -0
- data/app/forms/uffizzi_core/api/cli/v1/template/create_form.rb +65 -0
- data/app/forms/uffizzi_core/application_form.rb +11 -0
- data/app/forms/uffizzi_core/application_form_without_active_record.rb +17 -0
- data/app/forms/uffizzi_core/mass_assignment_control_concern.rb +22 -0
- data/app/helpers/uffizzi_core/application_helper.rb +6 -0
- data/app/jobs/uffizzi_core/account/create_credential_job.rb +10 -0
- data/app/jobs/uffizzi_core/activity_item/docker/update_digest_job.rb +11 -0
- data/app/jobs/uffizzi_core/application_job.rb +7 -0
- data/app/jobs/uffizzi_core/config_file/apply_job.rb +31 -0
- data/app/jobs/uffizzi_core/credential/docker_hub/create_webhook_job.rb +15 -0
- data/app/jobs/uffizzi_core/deployment/create_credential_job.rb +32 -0
- data/app/jobs/uffizzi_core/deployment/create_credentials_job.rb +17 -0
- data/app/jobs/uffizzi_core/deployment/create_job.rb +15 -0
- data/app/jobs/uffizzi_core/deployment/create_webhooks_job.rb +13 -0
- data/app/jobs/uffizzi_core/deployment/delete_credential_job.rb +13 -0
- data/app/jobs/uffizzi_core/deployment/delete_job.rb +11 -0
- data/app/jobs/uffizzi_core/deployment/deploy_containers_job.rb +27 -0
- data/app/jobs/uffizzi_core/deployment/manage_deploy_activity_item_job.rb +19 -0
- data/app/jobs/uffizzi_core/deployment/send_github_preview_message_job.rb +13 -0
- data/app/lib/uffizzi_core/rbac/user_access_service.rb +21 -0
- data/app/mailers/uffizzi_core/application_mailer.rb +8 -0
- data/app/models/concerns/uffizzi_core/hashid_concern.rb +25 -0
- data/app/models/concerns/uffizzi_core/state_machine_concern.rb +16 -0
- data/app/models/uffizzi_core/account.rb +101 -0
- data/app/models/uffizzi_core/activity_item/docker.rb +4 -0
- data/app/models/uffizzi_core/activity_item/github.rb +4 -0
- data/app/models/uffizzi_core/activity_item/memory_limit.rb +4 -0
- data/app/models/uffizzi_core/activity_item.rb +58 -0
- data/app/models/uffizzi_core/application_record.rb +7 -0
- data/app/models/uffizzi_core/build.rb +39 -0
- data/app/models/uffizzi_core/comment.rb +16 -0
- data/app/models/uffizzi_core/compose_file.rb +57 -0
- data/app/models/uffizzi_core/config_file.rb +24 -0
- data/app/models/uffizzi_core/container.rb +100 -0
- data/app/models/uffizzi_core/container_config_file.rb +8 -0
- data/app/models/uffizzi_core/continuous_preview.rb +4 -0
- data/app/models/uffizzi_core/coupon.rb +5 -0
- data/app/models/uffizzi_core/credential/amazon.rb +4 -0
- data/app/models/uffizzi_core/credential/azure.rb +4 -0
- data/app/models/uffizzi_core/credential/docker_hub.rb +4 -0
- data/app/models/uffizzi_core/credential/github.rb +4 -0
- data/app/models/uffizzi_core/credential/google.rb +4 -0
- data/app/models/uffizzi_core/credential.rb +64 -0
- data/app/models/uffizzi_core/database.rb +4 -0
- data/app/models/uffizzi_core/database_offering.rb +4 -0
- data/app/models/uffizzi_core/deployment.rb +77 -0
- data/app/models/uffizzi_core/event.rb +13 -0
- data/app/models/uffizzi_core/invitation.rb +27 -0
- data/app/models/uffizzi_core/membership.rb +16 -0
- data/app/models/uffizzi_core/payment.rb +11 -0
- data/app/models/uffizzi_core/price.rb +9 -0
- data/app/models/uffizzi_core/product.rb +11 -0
- data/app/models/uffizzi_core/project.rb +66 -0
- data/app/models/uffizzi_core/rating.rb +20 -0
- data/app/models/uffizzi_core/repo/amazon.rb +4 -0
- data/app/models/uffizzi_core/repo/azure.rb +4 -0
- data/app/models/uffizzi_core/repo/docker_hub.rb +4 -0
- data/app/models/uffizzi_core/repo/github.rb +4 -0
- data/app/models/uffizzi_core/repo/google.rb +4 -0
- data/app/models/uffizzi_core/repo.rb +39 -0
- data/app/models/uffizzi_core/role.rb +17 -0
- data/app/models/uffizzi_core/template.rb +19 -0
- data/app/models/uffizzi_core/user.rb +62 -0
- data/app/models/uffizzi_core/user_project.rb +14 -0
- data/app/policies/uffizzi_core/api/cli/v1/account/credentials_policy.rb +11 -0
- data/app/policies/uffizzi_core/api/cli/v1/projects/compose_files_policy.rb +15 -0
- data/app/policies/uffizzi_core/api/cli/v1/projects/deployments/activity_items_policy.rb +7 -0
- data/app/policies/uffizzi_core/api/cli/v1/projects/deployments/containers_policy.rb +7 -0
- data/app/policies/uffizzi_core/api/cli/v1/projects/deployments/events_policy.rb +7 -0
- data/app/policies/uffizzi_core/api/cli/v1/projects/deployments_policy.rb +23 -0
- data/app/policies/uffizzi_core/api/cli/v1/projects/secrets_policy.rb +15 -0
- data/app/policies/uffizzi_core/api/cli/v1/projects_policy.rb +7 -0
- data/app/policies/uffizzi_core/application_policy.rb +12 -0
- data/app/repositories/uffizzi_core/activity_item_repo.rb +9 -0
- data/app/repositories/uffizzi_core/basic_order_repo.rb +11 -0
- data/app/repositories/uffizzi_core/build_repo.rb +23 -0
- data/app/repositories/uffizzi_core/comment_repo.rb +11 -0
- data/app/repositories/uffizzi_core/compose_file_repo.rb +11 -0
- data/app/repositories/uffizzi_core/config_file_repo.rb +40 -0
- data/app/repositories/uffizzi_core/container_repo.rb +25 -0
- data/app/repositories/uffizzi_core/credential_repo.rb +36 -0
- data/app/repositories/uffizzi_core/deployment_repo.rb +23 -0
- data/app/repositories/uffizzi_core/event_repo.rb +9 -0
- data/app/repositories/uffizzi_core/membership_repo.rb +10 -0
- data/app/repositories/uffizzi_core/price_repo.rb +11 -0
- data/app/repositories/uffizzi_core/product_repo.rb +11 -0
- data/app/repositories/uffizzi_core/project_repo.rb +10 -0
- data/app/repositories/uffizzi_core/repo_repo.rb +10 -0
- data/app/repositories/uffizzi_core/template_repo.rb +87 -0
- data/app/repositories/uffizzi_core/usage_repo.rb +9 -0
- data/app/repositories/uffizzi_core/user_repo.rb +11 -0
- data/app/responders/uffizzi_core/json_responder.rb +13 -0
- data/app/serializers/uffizzi_core/api/cli/v1/account/credential_serializer.rb +9 -0
- data/app/serializers/uffizzi_core/api/cli/v1/project_serializer.rb +7 -0
- data/app/serializers/uffizzi_core/api/cli/v1/projects/compose_file_serializer.rb +7 -0
- data/app/serializers/uffizzi_core/api/cli/v1/projects/deployment_serializer/container_serializer.rb +23 -0
- data/app/serializers/uffizzi_core/api/cli/v1/projects/deployment_serializer/user_serializer.rb +11 -0
- data/app/serializers/uffizzi_core/api/cli/v1/projects/deployment_serializer.rb +74 -0
- data/app/serializers/uffizzi_core/api/cli/v1/projects/deployments/activity_item_serializer.rb +24 -0
- data/app/serializers/uffizzi_core/api/cli/v1/projects/deployments/container_serializer/container_config_file_serializer/config_file_serializer.rb +6 -0
- data/app/serializers/uffizzi_core/api/cli/v1/projects/deployments/container_serializer/container_config_file_serializer.rb +7 -0
- data/app/serializers/uffizzi_core/api/cli/v1/projects/deployments/container_serializer.rb +30 -0
- data/app/serializers/uffizzi_core/api/cli/v1/user_serializer/account_serializer.rb +5 -0
- data/app/serializers/uffizzi_core/api/cli/v1/user_serializer.rb +7 -0
- data/app/serializers/uffizzi_core/base_serializer.rb +7 -0
- data/app/serializers/uffizzi_core/controller/create_credential/credential_serializer.rb +17 -0
- data/app/serializers/uffizzi_core/controller/create_deployment/deployment_serializer.rb +5 -0
- data/app/serializers/uffizzi_core/controller/deploy_containers/container_serializer.rb +53 -0
- data/app/serializers/uffizzi_core/controller/deploy_containers/credential_serializer.rb +5 -0
- data/app/services/uffizzi_core/account_service.rb +21 -0
- data/app/services/uffizzi_core/activity_item_service.rb +98 -0
- data/app/services/uffizzi_core/amazon/credential_service.rb +31 -0
- data/app/services/uffizzi_core/amazon_service.rb +45 -0
- data/app/services/uffizzi_core/azure/credential_service.rb +18 -0
- data/app/services/uffizzi_core/cli/compose_file_service.rb +203 -0
- data/app/services/uffizzi_core/compose_file/builders/config_files_builder_service.rb +31 -0
- data/app/services/uffizzi_core/compose_file/builders/container_builder_service.rb +225 -0
- data/app/services/uffizzi_core/compose_file/builders/docker_repo_builder_service.rb +25 -0
- data/app/services/uffizzi_core/compose_file/builders/github_repo_builder_service.rb +59 -0
- data/app/services/uffizzi_core/compose_file/builders/template_builder_service.rb +45 -0
- data/app/services/uffizzi_core/compose_file/builders/variables_builder_service.rb +58 -0
- data/app/services/uffizzi_core/compose_file/config_files_service.rb +52 -0
- data/app/services/uffizzi_core/compose_file/config_option_service.rb +37 -0
- data/app/services/uffizzi_core/compose_file/configs_options_service.rb +26 -0
- data/app/services/uffizzi_core/compose_file/container_service.rb +64 -0
- data/app/services/uffizzi_core/compose_file/continuous_preview_options_service.rb +57 -0
- data/app/services/uffizzi_core/compose_file/dependencies_service.rb +55 -0
- data/app/services/uffizzi_core/compose_file/errors_service.rb +46 -0
- data/app/services/uffizzi_core/compose_file/github_dependencies_service.rb +38 -0
- data/app/services/uffizzi_core/compose_file/ingress_options_service.rb +49 -0
- data/app/services/uffizzi_core/compose_file/secrets_options_service.rb +28 -0
- data/app/services/uffizzi_core/compose_file/services_options/build_service.rb +93 -0
- data/app/services/uffizzi_core/compose_file/services_options/command_service.rb +18 -0
- data/app/services/uffizzi_core/compose_file/services_options/configs_service.rb +51 -0
- data/app/services/uffizzi_core/compose_file/services_options/deploy_service.rb +44 -0
- data/app/services/uffizzi_core/compose_file/services_options/entrypoint_service.rb +18 -0
- data/app/services/uffizzi_core/compose_file/services_options/env_file_service.rb +34 -0
- data/app/services/uffizzi_core/compose_file/services_options/environment_service.rb +20 -0
- data/app/services/uffizzi_core/compose_file/services_options/image_service.rb +89 -0
- data/app/services/uffizzi_core/compose_file/services_options/secrets_service.rb +35 -0
- data/app/services/uffizzi_core/compose_file/services_options_service.rb +55 -0
- data/app/services/uffizzi_core/compose_file/template_service.rb +55 -0
- data/app/services/uffizzi_core/compose_file/update_service.rb +29 -0
- data/app/services/uffizzi_core/compose_file/variables_service.rb +25 -0
- data/app/services/uffizzi_core/compose_file_service.rb +33 -0
- data/app/services/uffizzi_core/container_service.rb +57 -0
- data/app/services/uffizzi_core/controller_service.rb +80 -0
- data/app/services/uffizzi_core/credential_service.rb +44 -0
- data/app/services/uffizzi_core/deployment_service.rb +274 -0
- data/app/services/uffizzi_core/docker_hub/credential_service.rb +15 -0
- data/app/services/uffizzi_core/docker_hub_service.rb +77 -0
- data/app/services/uffizzi_core/github/app_service.rb +51 -0
- data/app/services/uffizzi_core/github/credential_service.rb +124 -0
- data/app/services/uffizzi_core/github/message_service.rb +20 -0
- data/app/services/uffizzi_core/github_service.rb +28 -0
- data/app/services/uffizzi_core/google/credential_service.rb +18 -0
- data/app/services/uffizzi_core/logs_service.rb +33 -0
- data/app/services/uffizzi_core/manage_activity_items_service.rb +166 -0
- data/app/services/uffizzi_core/project_service.rb +38 -0
- data/app/services/uffizzi_core/repo_service.rb +178 -0
- data/app/services/uffizzi_core/response_service.rb +13 -0
- data/app/services/uffizzi_core/template_service.rb +21 -0
- data/app/services/uffizzi_core/token_service.rb +19 -0
- data/app/services/uffizzi_core/user_access_service.rb +14 -0
- data/app/utils/uffizzi_core/converters.rb +33 -0
- data/app/validators/uffizzi_core/email_validator.rb +9 -0
- data/app/validators/uffizzi_core/environment_variable_list_validator.rb +15 -0
- data/app/validators/uffizzi_core/image_command_args_validator.rb +21 -0
- data/config/initializers/rswag_api.rb +15 -0
- data/config/initializers/rswag_ui.rb +15 -0
- data/config/initializers/swagger_yard.rb +17 -0
- data/config/locales/en.activerecord.yml +18 -0
- data/config/locales/en.yml +61 -0
- data/config/routes.rb +55 -0
- data/db/migrate/20220218121438_create_uffizzi_core_tables.rb +375 -0
- data/db/migrate/20220325113342_add_name_to_uffizzi_containers.rb +7 -0
- data/db/seeds.rb +16 -0
- data/lib/tasks/uffizzi_core_tasks.rake +14 -0
- data/lib/uffizzi_core/engine.rb +15 -0
- data/lib/uffizzi_core/version.rb +5 -0
- data/lib/uffizzi_core.rb +60 -0
- data/swagger/v1/swagger.json +1278 -0
- metadata +935 -0
@@ -0,0 +1,69 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# @resource Project/Secrets
|
4
|
+
class UffizziCore::Api::Cli::V1::Projects::SecretsController < UffizziCore::Api::Cli::V1::Projects::ApplicationController
|
5
|
+
before_action :authorize_uffizzi_core_api_cli_v1_projects_secrets
|
6
|
+
|
7
|
+
# Get secrets for the project
|
8
|
+
#
|
9
|
+
# @path [GET] /api/cli/v1/projects/{project_slug}/secrets
|
10
|
+
# @parameter project_slug(required,path) [string]
|
11
|
+
# @response [object<secrets: Array<object<name: string>> >] 200 OK
|
12
|
+
# @response 401 Not authorized
|
13
|
+
def index
|
14
|
+
project_secrets = resource_project.secrets.present? ? resource_project.secrets : []
|
15
|
+
secrets = project_secrets.map { |secret| { name: secret['name'] } }
|
16
|
+
|
17
|
+
render json: { secrets: secrets }, status: :ok
|
18
|
+
end
|
19
|
+
|
20
|
+
# Add secret to project
|
21
|
+
#
|
22
|
+
# @path [POST] /api/cli/v1/projects/{project_slug}/secrets/bulk_create
|
23
|
+
# @parameter project_slug(required,path) [string]
|
24
|
+
# @parameter secrets(required,body) [object<secrets: Array<object <name: string, value: string>>>]
|
25
|
+
# @response [object<secrets: Array<object<name: string>>>] 201 Created
|
26
|
+
# @response 422 A compose file already exists for this project
|
27
|
+
# @response 401 Not authorized
|
28
|
+
def bulk_create
|
29
|
+
project_form = resource_project.becomes(UffizziCore::Api::Cli::V1::Project::UpdateForm)
|
30
|
+
project_form.assign_secrets!(secrets_params)
|
31
|
+
return render json: { errors: project_form.errors }, status: :unprocessable_entity unless project_form.save
|
32
|
+
|
33
|
+
UffizziCore::ProjectService.update_compose_secrets(project_form)
|
34
|
+
secrets = project_form.secrets.map { |secret| { name: secret['name'] } }
|
35
|
+
|
36
|
+
render json: { secrets: secrets }, status: :created
|
37
|
+
end
|
38
|
+
|
39
|
+
# Delete a secret from project by secret id
|
40
|
+
#
|
41
|
+
# @path [DELETE] /api/cli/v1/projects/{project_slug}/secrets/{id}
|
42
|
+
# @parameter project_slug(required,path) [string]
|
43
|
+
# @response [Project] 200 OK
|
44
|
+
# @response 422
|
45
|
+
# @response 401 Not authorized
|
46
|
+
def destroy
|
47
|
+
secret_name = CGI.unescape(params[:id])
|
48
|
+
secret = OpenStruct.new(name: secret_name)
|
49
|
+
project_form = resource_project.becomes(UffizziCore::Api::Cli::V1::Project::DeleteSecretForm)
|
50
|
+
project_form.secret = secret
|
51
|
+
|
52
|
+
if project_form.invalid?
|
53
|
+
return respond_with project_form
|
54
|
+
end
|
55
|
+
|
56
|
+
project_form.delete_secret!
|
57
|
+
if project_form.save!(validate: false)
|
58
|
+
UffizziCore::ProjectService.update_compose_secret_errors(project_form, secret)
|
59
|
+
end
|
60
|
+
|
61
|
+
respond_with project_form
|
62
|
+
end
|
63
|
+
|
64
|
+
private
|
65
|
+
|
66
|
+
def secrets_params
|
67
|
+
params.require(:secrets)
|
68
|
+
end
|
69
|
+
end
|
@@ -0,0 +1,19 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# @resource Project
|
4
|
+
|
5
|
+
class UffizziCore::Api::Cli::V1::ProjectsController < UffizziCore::Api::Cli::V1::ApplicationController
|
6
|
+
before_action :authorize_uffizzi_core_api_cli_v1_projects
|
7
|
+
|
8
|
+
# Get projects of current user
|
9
|
+
#
|
10
|
+
# @path [GET] /api/cli/v1/projects
|
11
|
+
#
|
12
|
+
# @response [object<projects: Array<object<slug: string>> >] 200 OK
|
13
|
+
# @response 401 Not authorized
|
14
|
+
def index
|
15
|
+
projects = current_user.projects.active.order(updated_at: :desc)
|
16
|
+
|
17
|
+
respond_with projects
|
18
|
+
end
|
19
|
+
end
|
@@ -0,0 +1,43 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# @resource Uffizzi
|
4
|
+
|
5
|
+
class UffizziCore::Api::Cli::V1::SessionsController < UffizziCore::Api::Cli::V1::ApplicationController
|
6
|
+
skip_before_action :authenticate_request!, only: [:create]
|
7
|
+
|
8
|
+
# Create session
|
9
|
+
#
|
10
|
+
# @path [POST] /api/cli/v1/session
|
11
|
+
#
|
12
|
+
# @parameter user(required,body) [object<email: string, password: string >]
|
13
|
+
# @response [object<user: object<accounts: Array<object<id: integer, state: string>> >>] 201 Created successfully
|
14
|
+
# @response [object<errors: object<password: string >>] 422 Unprocessable entity
|
15
|
+
def create
|
16
|
+
session_form = UffizziCore::Api::Cli::V1::SessionCreateForm.new(session_params)
|
17
|
+
|
18
|
+
if session_form.valid?
|
19
|
+
sign_in(session_form.user)
|
20
|
+
|
21
|
+
return respond_with session_form.user
|
22
|
+
end
|
23
|
+
|
24
|
+
respond_with session_form
|
25
|
+
end
|
26
|
+
|
27
|
+
# Destroy session
|
28
|
+
#
|
29
|
+
# @path [DELETE] /api/cli/v1/session
|
30
|
+
#
|
31
|
+
# @response 204 No Content
|
32
|
+
def destroy
|
33
|
+
sign_out
|
34
|
+
|
35
|
+
head :no_content
|
36
|
+
end
|
37
|
+
|
38
|
+
private
|
39
|
+
|
40
|
+
def session_params
|
41
|
+
params.require(:user).permit(:email, :password)
|
42
|
+
end
|
43
|
+
end
|
@@ -0,0 +1,51 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::ApplicationController < ActionController::Base
|
4
|
+
include Pundit
|
5
|
+
include UffizziCore::ResponseService
|
6
|
+
include UffizziCore::AuthManagement
|
7
|
+
include UffizziCore::AuthorizationConcern
|
8
|
+
include UffizziCore::DependencyInjectionConcern
|
9
|
+
|
10
|
+
DEFAULT_PAGE = 1
|
11
|
+
DEFAULT_PER_PAGE = 20
|
12
|
+
|
13
|
+
protect_from_forgery with: :exception
|
14
|
+
rescue_from ActiveRecord::RecordNotFound, with: :render_not_found
|
15
|
+
|
16
|
+
before_action :authenticate_request!
|
17
|
+
skip_before_action :verify_authenticity_token
|
18
|
+
rescue_from ActiveRecord::RecordNotFound, with: :render_not_found
|
19
|
+
|
20
|
+
respond_to :json
|
21
|
+
|
22
|
+
def policy_context
|
23
|
+
UffizziCore::BaseContext.new(current_user, user_access_module, params)
|
24
|
+
end
|
25
|
+
|
26
|
+
def self.responder
|
27
|
+
UffizziCore::JsonResponder
|
28
|
+
end
|
29
|
+
|
30
|
+
def render_not_found
|
31
|
+
render json: { errors: { title: ['Resource Not Found'] } }, status: :not_found
|
32
|
+
end
|
33
|
+
|
34
|
+
def render_errors(errors)
|
35
|
+
json = { errors: errors }
|
36
|
+
|
37
|
+
render json: json, status: :unprocessable_entity
|
38
|
+
end
|
39
|
+
|
40
|
+
def q_param
|
41
|
+
params[:q] || ActionController::Parameters.new
|
42
|
+
end
|
43
|
+
|
44
|
+
def page
|
45
|
+
params[:page] || DEFAULT_PAGE
|
46
|
+
end
|
47
|
+
|
48
|
+
def per_page
|
49
|
+
params[:per_page] || DEFAULT_PER_PAGE
|
50
|
+
end
|
51
|
+
end
|
@@ -0,0 +1,26 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::Account::Credential::CreateForm < UffizziCore::Credential
|
4
|
+
include UffizziCore::ApplicationForm
|
5
|
+
|
6
|
+
permit :type, :registry_url, :username, :password
|
7
|
+
|
8
|
+
validates :password, presence: { message: :password_blank }
|
9
|
+
validate :check_registry_url, if: -> { errors[:password].empty? }
|
10
|
+
validate :check_credential_correctness, if: -> { errors[:password].empty? }
|
11
|
+
validate :credential_exists?, if: -> { errors[:password].empty? }
|
12
|
+
|
13
|
+
private
|
14
|
+
|
15
|
+
def check_registry_url
|
16
|
+
errors.add(:registry_url, :invalid_scheme) if URI.parse(registry_url).scheme.nil?
|
17
|
+
end
|
18
|
+
|
19
|
+
def check_credential_correctness
|
20
|
+
errors.add(:username, :incorrect) unless UffizziCore::CredentialService.correct_credentials?(self)
|
21
|
+
end
|
22
|
+
|
23
|
+
def credential_exists?
|
24
|
+
errors.add(:type, :exist) if account.credentials.where(type: type).exists?
|
25
|
+
end
|
26
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::ComposeFile::CheckCredentialsForm
|
4
|
+
include UffizziCore::ApplicationFormWithoutActiveRecord
|
5
|
+
|
6
|
+
attribute :compose_file
|
7
|
+
attribute :credentials
|
8
|
+
|
9
|
+
validate :check_containers_credentials
|
10
|
+
|
11
|
+
private
|
12
|
+
|
13
|
+
def check_containers_credentials
|
14
|
+
compose_content = Base64.decode64(compose_file.content)
|
15
|
+
compose_data = UffizziCore::Cli::ComposeFileService.parse(compose_content)
|
16
|
+
|
17
|
+
UffizziCore::Cli::ComposeFileService.containers_credentials(compose_data, credentials)
|
18
|
+
rescue UffizziCore::ComposeFile::CredentialError => e
|
19
|
+
errors.add(:credentials, e.message)
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,39 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::ComposeFile::CliForm
|
4
|
+
include UffizziCore::ApplicationFormWithoutActiveRecord
|
5
|
+
|
6
|
+
attribute :credential, UffizziCore::Credential
|
7
|
+
attribute :compose_content_data, Hash
|
8
|
+
attribute :compose_data, Hash
|
9
|
+
attribute :compose_dependencies, Array
|
10
|
+
attribute :compose_repositories, Array
|
11
|
+
attribute :content, String
|
12
|
+
|
13
|
+
validates :content, presence: true
|
14
|
+
|
15
|
+
validate :check_compose_parsed_data, if: -> { errors[:content].empty? }
|
16
|
+
validate :check_repositories, if: -> { credential.present? && errors[:content].empty? }
|
17
|
+
validate :check_branches, if: -> { credential.present? && errors[:content].empty? }
|
18
|
+
|
19
|
+
def check_compose_parsed_data
|
20
|
+
compose_content = Base64.decode64(content)
|
21
|
+
self.compose_data = UffizziCore::Cli::ComposeFileService.parse(compose_content)
|
22
|
+
rescue UffizziCore::ComposeFile::ParseError => e
|
23
|
+
errors.add(:content, e.message)
|
24
|
+
end
|
25
|
+
|
26
|
+
def check_repositories
|
27
|
+
self.compose_repositories = UffizziCore::Cli::ComposeFileService.load_repositories(compose_data, credential)
|
28
|
+
rescue UffizziCore::ComposeFile::NotFoundError => e
|
29
|
+
errors.add(:content, e.message)
|
30
|
+
end
|
31
|
+
|
32
|
+
def check_branches
|
33
|
+
return if compose_repositories.blank?
|
34
|
+
|
35
|
+
UffizziCore::Cli::ComposeFileService.check_github_branches(compose_data, compose_repositories, credential)
|
36
|
+
rescue UffizziCore::ComposeFile::NotFoundError => e
|
37
|
+
errors.add(:content, e.message)
|
38
|
+
end
|
39
|
+
end
|
@@ -0,0 +1,13 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::ComposeFile::CreateForm < UffizziCore::ComposeFile
|
4
|
+
include UffizziCore::ApplicationForm
|
5
|
+
|
6
|
+
permit :source, :path, :content
|
7
|
+
|
8
|
+
validates :source, presence: true
|
9
|
+
validates :source, uniqueness: { scope: :project_id, message: 'A compose file with the same source already exists for this project' },
|
10
|
+
if: -> { kind.main? }
|
11
|
+
validates :path, presence: true
|
12
|
+
validates :content, presence: true
|
13
|
+
end
|
@@ -0,0 +1,44 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::ComposeFile::TemplateForm
|
4
|
+
include UffizziCore::ApplicationFormWithoutActiveRecord
|
5
|
+
|
6
|
+
SECRETS_ERROR_KEY = 'secret_variables'
|
7
|
+
TEMPLATE_BUILD_ERROR_KEY = 'template_build_error'
|
8
|
+
|
9
|
+
attribute :credentials
|
10
|
+
attribute :project, UffizziCore::Project
|
11
|
+
attribute :user, UffizziCore::User
|
12
|
+
attribute :compose_data, Hash
|
13
|
+
attribute :source, String
|
14
|
+
attribute :template_attributes, Hash
|
15
|
+
attribute :template_build_error, String
|
16
|
+
attribute :compose_dependencies, Array
|
17
|
+
attribute :compose_repositories, Array
|
18
|
+
|
19
|
+
validate :check_template_attributes
|
20
|
+
|
21
|
+
def assign_template_attributes!
|
22
|
+
self.template_attributes = UffizziCore::Cli::ComposeFileService.build_template_attributes(
|
23
|
+
compose_data,
|
24
|
+
source,
|
25
|
+
credentials,
|
26
|
+
project,
|
27
|
+
compose_dependencies,
|
28
|
+
compose_repositories,
|
29
|
+
)
|
30
|
+
rescue StandardError => e
|
31
|
+
self.template_build_error = e
|
32
|
+
end
|
33
|
+
|
34
|
+
private
|
35
|
+
|
36
|
+
def check_template_attributes
|
37
|
+
case template_build_error
|
38
|
+
when UffizziCore::ComposeFile::SecretsError
|
39
|
+
errors.add(SECRETS_ERROR_KEY, template_build_error.message)
|
40
|
+
when UffizziCore::ComposeFile::BuildError
|
41
|
+
errors.add(TEMPLATE_BUILD_ERROR_KEY, template_build_error.message)
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
@@ -0,0 +1,11 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::ConfigFile::CreateForm < UffizziCore::ConfigFile
|
4
|
+
include UffizziCore::ApplicationForm
|
5
|
+
|
6
|
+
permit :filename, :kind, :payload
|
7
|
+
|
8
|
+
validates :filename, presence: true
|
9
|
+
validates :kind, presence: true
|
10
|
+
validates :payload, presence: true
|
11
|
+
end
|
@@ -0,0 +1,91 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::Deployment::CreateForm < UffizziCore::Deployment
|
4
|
+
include UffizziCore::ApplicationForm
|
5
|
+
|
6
|
+
permit :creation_source,
|
7
|
+
containers_attributes: [
|
8
|
+
:image,
|
9
|
+
:name,
|
10
|
+
:tag,
|
11
|
+
:port,
|
12
|
+
:public,
|
13
|
+
:memory_limit,
|
14
|
+
:memory_request,
|
15
|
+
:entrypoint,
|
16
|
+
:command,
|
17
|
+
:receive_incoming_requests,
|
18
|
+
:continuously_deploy,
|
19
|
+
{ variables: [:name, :value],
|
20
|
+
secret_variables: [:name, :value],
|
21
|
+
repo_attributes: [
|
22
|
+
:namespace,
|
23
|
+
:name,
|
24
|
+
:slug,
|
25
|
+
:type,
|
26
|
+
:description,
|
27
|
+
:is_private,
|
28
|
+
:repository_id,
|
29
|
+
:branch,
|
30
|
+
:kind,
|
31
|
+
:dockerfile_path,
|
32
|
+
:dockerfile_context_path,
|
33
|
+
:deploy_preview_when_pull_request_is_opened,
|
34
|
+
:delete_preview_when_pull_request_is_closed,
|
35
|
+
:deploy_preview_when_image_tag_is_created,
|
36
|
+
:delete_preview_when_image_tag_is_updated,
|
37
|
+
:share_to_github,
|
38
|
+
:delete_preview_after,
|
39
|
+
{ args: [:name, :value] },
|
40
|
+
],
|
41
|
+
container_config_files_attributes: [
|
42
|
+
:config_file_id,
|
43
|
+
:mount_path,
|
44
|
+
] },
|
45
|
+
]
|
46
|
+
|
47
|
+
validate :check_all_containers_have_unique_ports
|
48
|
+
validate :check_exists_ingress_container
|
49
|
+
validate :check_max_memory_limit
|
50
|
+
validate :check_max_memory_request
|
51
|
+
|
52
|
+
def assign_dependences!(project, user)
|
53
|
+
self.project = project
|
54
|
+
|
55
|
+
self.containers = containers.map do |container|
|
56
|
+
container.repo.project = project if !container.repo.nil?
|
57
|
+
|
58
|
+
container
|
59
|
+
end
|
60
|
+
|
61
|
+
self.deployed_by = user
|
62
|
+
|
63
|
+
self
|
64
|
+
end
|
65
|
+
|
66
|
+
private
|
67
|
+
|
68
|
+
def check_all_containers_have_unique_ports
|
69
|
+
active_containers = containers.select(&:active?)
|
70
|
+
|
71
|
+
errors.add(:containers, :duplicate_ports) unless UffizziCore::DeploymentService.all_containers_have_unique_ports?(active_containers)
|
72
|
+
end
|
73
|
+
|
74
|
+
def check_exists_ingress_container
|
75
|
+
active_containers = containers.select(&:active?)
|
76
|
+
|
77
|
+
errors.add(:containers, :incorrect_ingress_container) unless UffizziCore::DeploymentService.ingress_container?(active_containers)
|
78
|
+
end
|
79
|
+
|
80
|
+
def check_max_memory_limit
|
81
|
+
return if UffizziCore::DeploymentService.valid_containers_memory_limit?(self)
|
82
|
+
|
83
|
+
errors.add(:containers, :max_memory_limit_error, max: project.account.container_memory_limit)
|
84
|
+
end
|
85
|
+
|
86
|
+
def check_max_memory_request
|
87
|
+
return if UffizziCore::DeploymentService.valid_containers_memory_request?(self)
|
88
|
+
|
89
|
+
errors.add(:containers, :max_memory_request_error, max: project.account.container_memory_limit)
|
90
|
+
end
|
91
|
+
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::Project::DeleteSecretForm < UffizziCore::Project
|
4
|
+
include UffizziCore::ApplicationForm
|
5
|
+
|
6
|
+
attr_accessor :secret
|
7
|
+
|
8
|
+
permit secrets: [:name, :value]
|
9
|
+
|
10
|
+
validate :check_existence
|
11
|
+
|
12
|
+
def delete_secret!
|
13
|
+
existing_secrets = secrets.presence || []
|
14
|
+
|
15
|
+
self.secrets = existing_secrets.reject { |existing_secret| existing_secret['name'] == secret.name }
|
16
|
+
end
|
17
|
+
|
18
|
+
private
|
19
|
+
|
20
|
+
def check_existence
|
21
|
+
existing_secrets = secrets.presence || []
|
22
|
+
|
23
|
+
detected_secret = existing_secrets.detect { |existing_secret| existing_secret['name'] == secret.name }
|
24
|
+
|
25
|
+
errors.add(:secret, :not_found, name: secret.name) if detected_secret.nil?
|
26
|
+
end
|
27
|
+
end
|
@@ -0,0 +1,40 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::Project::UpdateForm < UffizziCore::Project
|
4
|
+
include UffizziCore::ApplicationForm
|
5
|
+
MAX_SECRET_KEY_LENGTH = 256
|
6
|
+
|
7
|
+
permit :name, :slug, :description, secrets: [:name, :value]
|
8
|
+
|
9
|
+
validates :name, presence: true, uniqueness: { scope: :account }
|
10
|
+
validates :slug, presence: true, uniqueness: true
|
11
|
+
|
12
|
+
validate :check_duplicates
|
13
|
+
validate :check_length
|
14
|
+
|
15
|
+
def assign_secrets!(new_secrets)
|
16
|
+
existing_secrets = secrets.presence || []
|
17
|
+
|
18
|
+
self.secrets = existing_secrets.union(new_secrets)
|
19
|
+
end
|
20
|
+
|
21
|
+
private
|
22
|
+
|
23
|
+
def check_duplicates
|
24
|
+
duplicates = []
|
25
|
+
groupped_secrets = secrets.group_by { |secret| secret['name'] }
|
26
|
+
groupped_secrets.each_pair do |key, value|
|
27
|
+
duplicates << key if value.size > 1
|
28
|
+
end
|
29
|
+
|
30
|
+
error_message = I18n.t('secrets.duplicates_exists', secrets: duplicates.join(', '))
|
31
|
+
errors.add(:secrets, error_message) if duplicates.present?
|
32
|
+
end
|
33
|
+
|
34
|
+
def check_length
|
35
|
+
secrets_with_invalid_key_length = secrets.select { |secret| secret['name'].length > MAX_SECRET_KEY_LENGTH }
|
36
|
+
|
37
|
+
error_message = I18n.t('secrets.invalid_key_length')
|
38
|
+
errors.add(:secrets, error_message) if secrets_with_invalid_key_length.present?
|
39
|
+
end
|
40
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::SessionCreateForm
|
4
|
+
include UffizziCore::ApplicationFormWithoutActiveRecord
|
5
|
+
|
6
|
+
attribute :email, String
|
7
|
+
attribute :password, String
|
8
|
+
|
9
|
+
validates :email, :password, presence: true
|
10
|
+
validate :check_authenticate, if: :email
|
11
|
+
|
12
|
+
def user
|
13
|
+
@user ||= UffizziCore::User.active.by_email(email).first
|
14
|
+
end
|
15
|
+
|
16
|
+
def check_authenticate
|
17
|
+
return unless wrong_email_or_password?
|
18
|
+
|
19
|
+
errors.add(:password, 'Email or password is incorrect.')
|
20
|
+
end
|
21
|
+
|
22
|
+
private
|
23
|
+
|
24
|
+
def wrong_email_or_password?
|
25
|
+
return true if user.nil?
|
26
|
+
|
27
|
+
!user.authenticate(password)
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,65 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class UffizziCore::Api::Cli::V1::Template::CreateForm < UffizziCore::Template
|
4
|
+
include UffizziCore::ApplicationForm
|
5
|
+
|
6
|
+
permit :name,
|
7
|
+
payload: {
|
8
|
+
containers_attributes: [
|
9
|
+
:image,
|
10
|
+
:tag,
|
11
|
+
:port,
|
12
|
+
:public,
|
13
|
+
:memory_limit,
|
14
|
+
:memory_request,
|
15
|
+
:entrypoint,
|
16
|
+
:command,
|
17
|
+
:receive_incoming_requests,
|
18
|
+
:continuously_deploy,
|
19
|
+
:name,
|
20
|
+
{ variables: [:name, :value],
|
21
|
+
secret_variables: [:name, :value],
|
22
|
+
repo_attributes: [
|
23
|
+
:namespace,
|
24
|
+
:name,
|
25
|
+
:slug,
|
26
|
+
:type,
|
27
|
+
:description,
|
28
|
+
:repository_id,
|
29
|
+
:is_private,
|
30
|
+
:branch,
|
31
|
+
:kind,
|
32
|
+
:dockerfile_path,
|
33
|
+
:dockerfile_context_path,
|
34
|
+
:deploy_preview_when_pull_request_is_opened,
|
35
|
+
:delete_preview_when_pull_request_is_closed,
|
36
|
+
:deploy_preview_when_image_tag_is_created,
|
37
|
+
:delete_preview_when_image_tag_is_updated,
|
38
|
+
:share_to_github,
|
39
|
+
:delete_preview_after,
|
40
|
+
{ args: [:name, :value] },
|
41
|
+
],
|
42
|
+
container_config_files_attributes: [
|
43
|
+
:config_file_id,
|
44
|
+
:mount_path,
|
45
|
+
] },
|
46
|
+
],
|
47
|
+
}
|
48
|
+
|
49
|
+
validate :check_max_memory_limit
|
50
|
+
validate :check_max_memory_request
|
51
|
+
|
52
|
+
private
|
53
|
+
|
54
|
+
def check_max_memory_limit
|
55
|
+
return if UffizziCore::TemplateService.valid_containers_memory_limit?(self)
|
56
|
+
|
57
|
+
errors.add(:payload, :max_memory_limit_error, max: project.account.container_memory_limit)
|
58
|
+
end
|
59
|
+
|
60
|
+
def check_max_memory_request
|
61
|
+
return if UffizziCore::TemplateService.valid_containers_memory_request?(self)
|
62
|
+
|
63
|
+
errors.add(:payload, :max_memory_request_error, max: project.account.container_memory_limit)
|
64
|
+
end
|
65
|
+
end
|
@@ -0,0 +1,17 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module UffizziCore::ApplicationFormWithoutActiveRecord
|
4
|
+
extend ActiveSupport::Concern
|
5
|
+
|
6
|
+
included do
|
7
|
+
include ActiveModel::Validations
|
8
|
+
include ActiveModel::Conversion
|
9
|
+
include ActiveModel::Serialization
|
10
|
+
include ActiveModel::Validations::Callbacks
|
11
|
+
include ::Virtus.model
|
12
|
+
end
|
13
|
+
|
14
|
+
def persisted?
|
15
|
+
false
|
16
|
+
end
|
17
|
+
end
|