files.rb 4.6 KB
Newer Older
1 2
module API
  class Files < Grape::API
3 4 5 6
    helpers do
      def commit_params(attrs)
        {
          file_path: attrs[:file_path],
7
          start_branch: attrs[:branch],
8
          branch_name: attrs[:branch],
9 10
          commit_message: attrs[:commit_message],
          file_content: attrs[:content],
11 12 13
          file_content_encoding: attrs[:encoding],
          author_email: attrs[:author_email],
          author_name: attrs[:author_name]
14 15 16
        }
      end

17
      def assign_file_vars!
18 19
        authorize! :download_code, user_project

20 21 22
        @commit = user_project.commit(params[:ref])
        not_found!('Commit') unless @commit

23 24
        @repo = user_project.repository
        @blob = @repo.blob_at(@commit.sha, params[:file_path])
25 26 27 28 29

        not_found!('File') unless @blob
        @blob.load_all_data!(@repo)
      end

30 31 32
      def commit_response(attrs)
        {
          file_path: attrs[:file_path],
33
          branch: attrs[:branch]
34 35
        }
      end
Robert Schilling's avatar
Robert Schilling committed
36 37

      params :simple_file_params do
38
        requires :file_path, type: String, desc: 'The url encoded path to the file. Ex. lib%2Fclass%2Erb'
39
        requires :branch, type: String, desc: 'The name of branch'
Robert Schilling's avatar
Robert Schilling committed
40 41 42 43 44 45 46 47 48 49
        requires :commit_message, type: String, desc: 'Commit Message'
        optional :author_email, type: String, desc: 'The email of the author'
        optional :author_name, type: String, desc: 'The name of the author'
      end

      params :extended_file_params do
        use :simple_file_params
        requires :content, type: String, desc: 'File content'
        optional :encoding, type: String, values: %w[base64], desc: 'File encoding'
      end
50 51
    end

Robert Schilling's avatar
Robert Schilling committed
52 53 54
    params do
      requires :id, type: String, desc: 'The project ID'
    end
55
    resource :projects, requirements: { id: %r{[^/]+} } do
56
      desc 'Get raw file contents from the repository'
Robert Schilling's avatar
Robert Schilling committed
57
      params do
58 59
        requires :file_path, type: String, desc: 'The url encoded path to the file. Ex. lib%2Fclass%2Erb'
        requires :ref, type: String, desc: 'The name of branch, tag commit'
Robert Schilling's avatar
Robert Schilling committed
60
      end
61 62 63 64 65 66
      get ":id/repository/files/:file_path/raw" do
        assign_file_vars!

        send_git_blob @repo, @blob
      end

67
      desc 'Get a file from the repository'
68
      params do
69 70
        requires :file_path, type: String, desc: 'The url encoded path to the file. Ex. lib%2Fclass%2Erb'
        requires :ref, type: String, desc: 'The name of branch, tag or commit'
71
      end
72
      get ":id/repository/files/:file_path", requirements: { file_path: /.+/ } do
73
        assign_file_vars!
74

Robert Schilling's avatar
Robert Schilling committed
75
        {
76 77 78
          file_name: @blob.name,
          file_path: @blob.path,
          size: @blob.size,
Robert Schilling's avatar
Robert Schilling committed
79
          encoding: "base64",
80
          content: Base64.strict_encode64(@blob.data),
Robert Schilling's avatar
Robert Schilling committed
81
          ref: params[:ref],
82 83
          blob_id: @blob.id,
          commit_id: @commit.id,
84
          last_commit_id: @repo.last_commit_id_for_path(@commit.sha, params[:file_path])
Robert Schilling's avatar
Robert Schilling committed
85
        }
86 87
      end

Robert Schilling's avatar
Robert Schilling committed
88 89 90 91
      desc 'Create new file in repository'
      params do
        use :extended_file_params
      end
92
      post ":id/repository/files/:file_path", requirements: { file_path: /.+/ } do
93 94
        authorize! :push_code, user_project

Robert Schilling's avatar
Robert Schilling committed
95 96
        file_params = declared_params(include_missing: false)
        result = ::Files::CreateService.new(user_project, current_user, commit_params(file_params)).execute
97 98 99

        if result[:status] == :success
          status(201)
Robert Schilling's avatar
Robert Schilling committed
100
          commit_response(file_params)
101
        else
102
          render_api_error!(result[:message], 400)
103 104
        end
      end
105

Robert Schilling's avatar
Robert Schilling committed
106 107 108 109
      desc 'Update existing file in repository'
      params do
        use :extended_file_params
      end
110
      put ":id/repository/files/:file_path", requirements: { file_path: /.+/ } do
111 112
        authorize! :push_code, user_project

Robert Schilling's avatar
Robert Schilling committed
113 114
        file_params = declared_params(include_missing: false)
        result = ::Files::UpdateService.new(user_project, current_user, commit_params(file_params)).execute
115 116 117

        if result[:status] == :success
          status(200)
Robert Schilling's avatar
Robert Schilling committed
118
          commit_response(file_params)
119
        else
120 121
          http_status = result[:http_status] || 400
          render_api_error!(result[:message], http_status)
122 123
        end
      end
124

Robert Schilling's avatar
Robert Schilling committed
125 126 127 128
      desc 'Delete an existing file in repository'
      params do
        use :simple_file_params
      end
129
      delete ":id/repository/files/:file_path", requirements: { file_path: /.+/ } do
130 131
        authorize! :push_code, user_project

Robert Schilling's avatar
Robert Schilling committed
132
        file_params = declared_params(include_missing: false)
133
        result = ::Files::DeleteService.new(user_project, current_user, commit_params(file_params)).execute
134

135
        if result[:status] != :success
136
          render_api_error!(result[:message], 400)
137 138
        end
      end
139 140 141
    end
  end
end