| @@ -440,6 +440,9 @@ http://localhost:3000/api/projects/migrate | jq | |||
| |repository_name |是|string |仓库名称, 只含有数字、字母、下划线不能以下划线开头和结尾,且唯一 | | |||
| |project_category_id|是|int |项目类别id | | |||
| |project_language_id|是|int |项目语言id | | |||
| |is_mirror |否|boolean|是否设置为镜像, true:是, false:否,默认为否 | | |||
| |auth_username |否|string|镜像源仓库的登录用户名 | | |||
| |auth_password |否|string|镜像源仓库的登录秘密 | | |||
| |private |否|boolean|项目是否私有, true:为私有,false: 非私有,默认为公开 | | |||
| @@ -458,6 +461,49 @@ http://localhost:3000/api/projects/migrate | jq | |||
| "name": "ni项目" | |||
| } | |||
| ``` | |||
| #### 手动同步镜像 | |||
| ``` | |||
| POST api/repositories/:id/sync_mirror | |||
| ``` | |||
| *示例* | |||
| ``` | |||
| curl -X POST \ | |||
| -d "user_id=36401" \ | |||
| http://localhost:3000/api/repositories/1244/sync_mirror | jq | |||
| ``` | |||
| *请求参数说明:* | |||
| |参数名|必选|类型|说明| | |||
| |-|-|-|-| | |||
| |user_id |是|int |用户id或者组织id | | |||
| |name |是|string |项目名称 | | |||
| |clone_addr |是|string |镜像项目clone地址 | | |||
| |description |否|string |项目描述 | | |||
| |repository_name |是|string |仓库名称, 只含有数字、字母、下划线不能以下划线开头和结尾,且唯一 | | |||
| |project_category_id|是|int |项目类别id | | |||
| |project_language_id|是|int |项目语言id | | |||
| |is_mirror |否|boolean|是否设置为镜像, true:是, false:否,默认为否 | | |||
| |auth_username |否|string|镜像源仓库的登录用户名 | | |||
| |auth_password |否|string|镜像源仓库的登录秘密 | | |||
| |private |否|boolean|项目是否私有, true:为私有,false: 非私有,默认为公开 | | |||
| *返回参数说明:* | |||
| |参数名|类型|说明| | |||
| |-|-|-| | |||
| |status |int |状态码, 0:标识请求成功 | | |||
| |message |string|服务端返回的信息说明| | |||
| 返回值 | |||
| ``` | |||
| { | |||
| "status": 0, | |||
| "message": "success" | |||
| } | |||
| ``` | |||
| --- | |||
| #### 项目详情 | |||
| @@ -5,7 +5,7 @@ module OperateProjectAbilityAble | |||
| end | |||
| def authorizate_user_can_edit_project! | |||
| return if current_user.project_manager? @project || current_user.admin? | |||
| return if @project.manager?(current_user) || current_user.admin? | |||
| render_forbidden('你没有权限操作.') | |||
| end | |||
| @@ -17,7 +17,7 @@ class ProjectsController < ApplicationController | |||
| ActiveRecord::Base.transaction do | |||
| Projects::CreateForm.new(project_params).validate! | |||
| @project = Projects::CreateService.new(current_user, project_params).call | |||
| end | |||
| rescue Exception => e | |||
| uid_logger_error(e.message) | |||
| @@ -107,8 +107,8 @@ class ProjectsController < ApplicationController | |||
| end | |||
| def mirror_params | |||
| params.permit(:user_id, :name, :description, :repository_name, | |||
| :project_category_id, :project_language_id, :clone_addr, :private) | |||
| params.permit(:user_id, :name, :description, :repository_name, :is_mirror, :auth_username, | |||
| :auth_password, :project_category_id, :project_language_id, :clone_addr, :private) | |||
| end | |||
| def project_public? | |||
| @@ -1,7 +1,10 @@ | |||
| class RepositoriesController < ApplicationController | |||
| include ApplicationHelper | |||
| before_action :require_login, only: %i[edit update create_file update_file delete_file] | |||
| include OperateProjectAbilityAble | |||
| before_action :require_login, only: %i[edit update create_file update_file delete_file sync_mirror] | |||
| before_action :find_project, :authorizate! | |||
| before_action :find_repository, only: %i[sync_mirror] | |||
| before_action :authorizate_user_can_edit_project!, only: %i[sync_mirror] | |||
| def show | |||
| @branches_count = Gitea::Repository::BranchesService.new(@project.owner, @project.identifier).call&.size | |||
| @@ -125,8 +128,8 @@ class RepositoriesController < ApplicationController | |||
| def repo_hook | |||
| hook_type = request.headers["X-Gitea-Event"].to_s # 获取推送的方式 | |||
| ownername = @project.owner.try(:login) | |||
| reponame = @project.identifier | |||
| ownername = @project.owner.try(:login) | |||
| reponame = @project.identifier | |||
| username = current_user.try(:login) | |||
| user_params = { | |||
| "ownername": ownername, | |||
| @@ -142,6 +145,12 @@ class RepositoriesController < ApplicationController | |||
| @project.update_attribute(:token, @project.token + uploadPushInfo[:modificationLines].to_i) | |||
| end | |||
| def sync_mirror | |||
| @repo&.mirror.set_status!(Mirror.statuses[:waiting]) | |||
| SyncMirroredRepositoryJob(@repo, current_user) | |||
| render_ok | |||
| end | |||
| private | |||
| def find_project | |||
| @@ -149,6 +158,10 @@ class RepositoriesController < ApplicationController | |||
| render_not_found("未找到相关的仓库") unless @project | |||
| end | |||
| def find_repository | |||
| @repo = Repository.find params[:id] | |||
| end | |||
| def authorizate! | |||
| if @project.repository.hidden? && !@project.member?(current_user) | |||
| render_forbidden | |||
| @@ -166,13 +179,13 @@ class RepositoriesController < ApplicationController | |||
| } | |||
| end | |||
| def hook_params(hook_type, params) | |||
| def hook_params(hook_type, params) | |||
| if hook_type == "push" | |||
| # TODO hook返回的记录中,暂时没有文件代码数量的增减,暂时根据 commits数量来计算 | |||
| # TODO hook返回的记录中,暂时没有文件代码数量的增减,暂时根据 commits数量来计算 | |||
| uploadPushInfo = { | |||
| "shas": params["commits"].present? ? params["commits"].map{|c| c["id"]} : "", | |||
| "shas": params["commits"].present? ? params["commits"].map{|c| c["id"]} : "", | |||
| "branch": params["ref"].to_s.split("/").last, | |||
| "modificationLines": params["commits"].length | |||
| "modificationLines": params["commits"].length | |||
| } | |||
| elsif hook_type == "pull_request" && params["action"].to_s == "closed" #合并请求合并后才会有上链操作 | |||
| uploadPushInfo = { | |||
| @@ -183,7 +196,7 @@ class RepositoriesController < ApplicationController | |||
| "shas": [params["pull_request"]["merge_commit_sha"], params["pull_request"]["merge_base"]], | |||
| "modificationLines": 1 #pull_request中没有commits数量 | |||
| } | |||
| else | |||
| else | |||
| uploadPushInfo = {} | |||
| end | |||
| @@ -2,7 +2,7 @@ class Projects::MigrateForm < BaseForm | |||
| REPOSITORY_NAME_REGEX = /^(?!_)(?!.*?_$)[a-zA-Z0-9_-]+$/ #只含有数字、字母、下划线不能以下划线开头和结尾 | |||
| URL_REGEX = /\A(?:(?:https?|ftp):\/\/)(?:\S+(?::\S*)?@)?(?:(?!10(?:\.\d{1,3}){3})(?!127(?:\.\d{1,3}){3})(?!169\.254(?:\.\d{1,3}){2})(?!192\.168(?:\.\d{1,3}){2})(?!172\.(?:1[6-9]|2\d|3[0-1])(?:\.\d{1,3}){2})(?:[1-9]\d?|1\d\d|2[01]\d|22[0-3])(?:\.(?:1?\d{1,2}|2[0-4]\d|25[0-5])){2}(?:\.(?:[1-9]\d?|1\d\d|2[0-4]\d|25[0-4]))|(?:(?:[a-z\u00a1-\uffff0-9]+-?)*[a-z\u00a1-\uffff0-9]+)(?:\.(?:[a-z\u00a1-\uffff0-9]+-?)*[a-z\u00a1-\uffff0-9]+)*(?:\.(?:[a-z\u00a1-\uffff]{2,})))(?::\d{2,5})?(?:\/[^\s]*)?\z/i | |||
| attr_accessor :user_id, :name, :description, :repository_name, :project_category_id, :project_language_id, :clone_addr, :private | |||
| attr_accessor :user_id, :name, :description, :repository_name, :project_category_id, :project_language_id, :clone_addr, :private, :is_mirror, :auth_username, :auth_password | |||
| validates :user_id, :name, :description,:repository_name, :project_category_id, :project_language_id, presence: true | |||
| validates :repository_name, format: { with: REPOSITORY_NAME_REGEX, multiline: true, message: "只能含有数字、字母、下划线且不能以下划线开头和结尾" } | |||
| @@ -0,0 +1,18 @@ | |||
| class MigrateRemoteRepositoryJob < ApplicationJob | |||
| queue_as :default | |||
| def perform(repo, token, params) | |||
| gitea_repository = Gitea::Repository::MigrateService.new(token, params).call | |||
| sync_project(repo, gitea_repository) | |||
| sync_repository(repo, gitea_repository) | |||
| end | |||
| private | |||
| def sync_project(repo, gitea_repository) | |||
| repo&.project.update_columns(gpid: gitea_repository["id"], identifier: gitea_repository["name"]) if gitea_repository | |||
| end | |||
| def sync_repository(repository, gitea_repository) | |||
| repository.mirror.update_columns(statuses: Mirror.statuses[:succeeded]) if gitea_repository | |||
| end | |||
| end | |||
| @@ -0,0 +1,8 @@ | |||
| class SyncMirroredRepositoryJob < ApplicationJob | |||
| queue_as :default | |||
| def perform(repo, current_user) | |||
| result = Gitea::Repository::SyncMirroredService.new(repo.user.login, repo.identifier, token: current_user.gitea_token).call | |||
| repo&.mirror.set_status! if result[:status] === 200 | |||
| end | |||
| end | |||
| @@ -4,7 +4,9 @@ module ProjectOperable | |||
| included do | |||
| has_many :members, dependent: :destroy | |||
| has_many :except_owner_members, -> { joins(:roles).where.not(roles: { name: 'Manager' }) }, class_name: 'Member' | |||
| has_many :manager_members, -> { joins(:roles).where(roles: { name: 'Manager' }) }, class_name: 'Member' | |||
| has_many :managers, -> { joins(:roles).where(roles: { name: 'Manager' }) }, class_name: 'Member' | |||
| has_many :developers, -> { joins(:roles).where(roles: { name: 'Developer' }) }, class_name: 'Member' | |||
| has_many :reporters, -> { joins(:roles).where(roles: { name: 'Reporter' }) }, class_name: 'Member' | |||
| end | |||
| def add_member!(user_id, role_name='Developer') | |||
| @@ -35,6 +37,20 @@ module ProjectOperable | |||
| self.owner == user | |||
| end | |||
| # 项目管理员(包含项目拥有者),权限:仓库设置、仓库可读可写 | |||
| def manager?(user) | |||
| managers.exists? user | |||
| end | |||
| # 项目开发者,可读可写权限 | |||
| def develper?(user) | |||
| developers.exists? user | |||
| end | |||
| # 报告者,只有可读权限 | |||
| def reporter?(user) | |||
| reporters.exists? user | |||
| end | |||
| def set_developer_role(member) | |||
| role = Role.find_by_name 'Developer' | |||
| member.member_roles.create!(role: role) | |||
| @@ -0,0 +1,13 @@ | |||
| class Mirror < ApplicationRecord | |||
| # 0 - succeeded, 1 - waiting, 2 - failed | |||
| # 0: 同步镜像成功;1: 正在同步镜像;2: 同步失败,默认值为0 | |||
| enum status: { succeeded: 0, waiting: 1, failed: 2 } | |||
| belongs_to :repository | |||
| def set_status!(status=Mirror.statuses[:succeeded]) | |||
| update_column(status: status) | |||
| end | |||
| end | |||
| @@ -4,7 +4,11 @@ class Project < ApplicationRecord | |||
| include Watchable | |||
| include ProjectOperable | |||
| enum project_type: { mirror: 1, common: 0 } # common:开源托管项目, mirror:开源镜像项目 | |||
| # common:开源托管项目 | |||
| # mirror:普通镜像项目,没有定时同步功能 | |||
| # sync_mirror:同步镜像项目,有系统定时同步功能,且用户可手动同步操作 | |||
| # | |||
| enum project_type: { sync_mirror: 2, mirror: 1, common: 0 } | |||
| belongs_to :ignore, optional: true | |||
| belongs_to :license, optional: true | |||
| @@ -124,8 +128,8 @@ class Project < ApplicationRecord | |||
| def releases_size(current_user_id, type) | |||
| if current_user_id == self.user_id && type.to_s == "all" | |||
| self.repository.version_releases_count | |||
| else | |||
| self.repository.version_releases_count | |||
| else | |||
| self.repository.version_releases.releases_size | |||
| end | |||
| end | |||
| @@ -2,6 +2,7 @@ class Repository < ApplicationRecord | |||
| self.inheritance_column = nil # FIX The single-table inheritance mechanism failed | |||
| belongs_to :project, :touch => true | |||
| belongs_to :user | |||
| has_one :mirror, foreign_key: :repo_id | |||
| has_many :version_releases, dependent: :destroy | |||
| validates :identifier, presence: true | |||
| @@ -9,4 +10,9 @@ class Repository < ApplicationRecord | |||
| def to_param | |||
| self.identifier.parameterize | |||
| end | |||
| # with repository is mirror | |||
| def set_mirror! | |||
| self.build_mirror(status: Mirror.statuses[:waiting]).save | |||
| end | |||
| end | |||
| @@ -143,6 +143,8 @@ class Gitea::ClientService < ApplicationService | |||
| when 409 | |||
| message = "创建失败,请检查该分支合并是否已存在" | |||
| raise Error, mark + message | |||
| when 403 | |||
| {status: 403, message: '你没有权限操作!'} | |||
| else | |||
| if response&.body.blank? | |||
| message = "请求失败" | |||
| @@ -0,0 +1,31 @@ | |||
| class Gitea::Repository::GetByIdService < Gitea::ClientService | |||
| attr_reader :owner, :repo_id | |||
| def initialize(owner, repo_id) | |||
| @owner = owner | |||
| @repo_id = repo_id | |||
| end | |||
| def call | |||
| response = get(url, params) | |||
| render_result(response) | |||
| end | |||
| private | |||
| def params | |||
| Hash.new.merge(token: owner.gitea_token) | |||
| end | |||
| def url | |||
| "/repositories/#{repo_id}".freeze | |||
| end | |||
| def render_result(response) | |||
| case response.status | |||
| when 200 | |||
| JSON.parse(response.body) | |||
| else | |||
| nil | |||
| end | |||
| end | |||
| end | |||
| @@ -0,0 +1,30 @@ | |||
| # Sync a mirrored repository | |||
| class Gitea::Repository::SyncMirroredService < Gitea::ClientService | |||
| attr_reader :token, :owner, :repo | |||
| # owner * | |||
| # owner of the repo to sync | |||
| # repo * | |||
| # name of the repo to sync | |||
| # example: | |||
| # Gitea::Repository::SyncMirroredService.new(owner.login, repo.identifier, user.gitea_token).call | |||
| def initialize(owner, repo, token=nil) | |||
| @token = token | |||
| @owner = owner | |||
| @repo = repo | |||
| end | |||
| def call | |||
| post(url, request_params) | |||
| end | |||
| private | |||
| def request_params | |||
| Hash.new.merge(token: token) | |||
| end | |||
| def url | |||
| "/repos/#{owner}/#{repo}/mirror-sync".freeze | |||
| end | |||
| end | |||
| @@ -31,17 +31,23 @@ class Projects::MigrateService < ApplicationService | |||
| project_category_id: params[:project_category_id], | |||
| project_language_id: params[:project_language_id], | |||
| is_public: project_secretion[:public], | |||
| project_type: Project.project_types[:mirror] | |||
| project_type: set_project_type | |||
| } | |||
| end | |||
| def set_project_type | |||
| ActiveModel::Type::Boolean.new.cast(params[:is_mirror]) == true ? Project.project_types[:sync_mirror] : Project.project_types[:mirror] | |||
| end | |||
| def repository_params | |||
| { | |||
| hidden: project_secretion[:hidden], | |||
| identifier: params[:repository_name], | |||
| mirror_url: params[:clone_addr], | |||
| user_id: user.id, | |||
| login: user.login | |||
| login: params[:auth_username], | |||
| password: params[:auth_password], | |||
| is_mirror: params[:is_mirror] | |||
| } | |||
| end | |||
| @@ -11,9 +11,8 @@ class Repositories::MigrateService < ApplicationService | |||
| @repository = Repository.new(repository_params) | |||
| ActiveRecord::Base.transaction do | |||
| if @repository.save! | |||
| gitea_repository = Gitea::Repository::MigrateService.new(user.gitea_token, gitea_repository_params).call | |||
| sync_project(gitea_repository) | |||
| sync_repository(@repository, gitea_repository) | |||
| @repository.set_mirror! if wrapper_mirror | |||
| MigrateRemoteRepositoryJob.perform_later(@repository, user.gitea_token, gitea_repository_params) | |||
| end | |||
| @repository | |||
| end | |||
| @@ -23,15 +22,6 @@ class Repositories::MigrateService < ApplicationService | |||
| end | |||
| private | |||
| def sync_project(gitea_repository) | |||
| project.update_columns(gpid: gitea_repository["id"], identifier: gitea_repository["name"]) if gitea_repository | |||
| end | |||
| def sync_repository(repository, gitea_repository) | |||
| repository.update_columns(url: gitea_repository["clone_url"]) if gitea_repository | |||
| end | |||
| def repository_params | |||
| params.merge(project_id: project.id) | |||
| end | |||
| @@ -41,7 +31,14 @@ class Repositories::MigrateService < ApplicationService | |||
| clone_addr: params[:mirror_url], | |||
| repo_name: params[:identifier], | |||
| uid: user.gitea_uid, | |||
| private: params[:hidden] | |||
| private: params[:hidden], | |||
| mirror: wrapper_mirror || false, | |||
| auth_username: params[:login], | |||
| auth_password: params[:password] | |||
| } | |||
| end | |||
| def wrapper_mirror | |||
| ActiveModel::Type::Boolean.new.cast(params[:is_mirror]) | |||
| end | |||
| end | |||
| @@ -197,6 +197,7 @@ Rails.application.routes.draw do | |||
| put :update_file | |||
| delete :delete_file | |||
| post :repo_hook | |||
| post :sync_mirror | |||
| end | |||
| end | |||
| @@ -0,0 +1,5 @@ | |||
| class AddForkUrlToRepositories < ActiveRecord::Migration[5.2] | |||
| def change | |||
| add_column :repositories, :fork_url, :string, default: nil | |||
| end | |||
| end | |||
| @@ -0,0 +1,12 @@ | |||
| class CreateMirrors < ActiveRecord::Migration[5.2] | |||
| def change | |||
| create_table :mirrors do |t| | |||
| t.integer :repo_id, foreign_key: true | |||
| t.integer :status, default: 0, null: false, comment: "0 - succeeded, 1 - waiting, 2 - failed" | |||
| t.integer :interval, comment: "mirror interval with unix time" | |||
| t.datetime :next_update_time, comment: "next update mirror time, for datetime" | |||
| t.timestamps | |||
| end | |||
| end | |||
| end | |||
| @@ -0,0 +1,5 @@ | |||
| class AddIsMirrorToRepositories < ActiveRecord::Migration[5.2] | |||
| def change | |||
| add_column :repositories, :is_mirror, :boolean, default: false | |||
| end | |||
| end | |||