Patch #1783 ยป autorepo.diff
| app/apis/sys_api.rb | ||
|---|---|---|
| 15 | 15 |
# along with this program; if not, write to the Free Software |
| 16 | 16 |
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. |
| 17 | 17 | |
| 18 |
class ResultElement < ActionWebService::Struct |
|
| 19 |
member :project, Project |
|
| 20 |
member :repository, Repository |
|
| 21 |
end |
|
| 22 | ||
| 18 | 23 |
class SysApi < ActionWebService::API::Base |
| 19 | 24 |
api_method :projects, |
| 20 | 25 |
:expects => [], |
| 21 |
:returns => [[Project]]
|
|
| 26 |
:returns => [[ResultElement]],
|
|
| 22 | 27 |
api_method :repository_created, |
| 23 | 28 |
:expects => [:string, :string], |
| 24 | 29 |
:returns => [:int] |
| app/controllers/repositories_controller.rb | ||
|---|---|---|
| 35 | 35 |
if !@repository |
| 36 | 36 |
@repository = Repository.factory(params[:repository_scm]) |
| 37 | 37 |
@repository.project = @project |
| 38 |
@repository.auto = params[:repository_auto] |
|
| 38 | 39 |
end |
| 39 | 40 |
if request.post? |
| 40 | 41 |
@repository.attributes = params[:repository] |
| app/controllers/sys_controller.rb | ||
|---|---|---|
| 23 | 23 |
before_invocation :check_enabled |
| 24 | 24 |
|
| 25 | 25 |
# Returns the projects list, with their repositories |
| 26 |
# Bug ! the list of repositories is not returned |
|
| 26 | 27 |
def projects |
| 27 |
Project.find(:all, :include => :repository) |
|
| 28 |
projects = Project.find(:all, :include => :repository) |
|
| 29 |
resultArray = [] |
|
| 30 |
projects.each do |project| |
|
| 31 |
resultElement = ResultElement.new |
|
| 32 |
resultElement.project = project |
|
| 33 |
resultElement.repository = project.repository |
|
| 34 |
resultArray << resultElement |
|
| 35 |
end |
|
| 36 |
resultArray |
|
| 28 | 37 |
end |
| 29 | 38 | |
| 30 | 39 |
# Registers a repository for the given project identifier |
| 31 |
# (Subversion specific) |
|
| 32 | 40 |
def repository_created(identifier, url) |
| 33 | 41 |
project = Project.find_by_identifier(identifier) |
| 34 |
# Do not create the repository if the project has already one |
|
| 35 |
return 0 unless project && project.repository.nil? |
|
| 36 |
logger.debug "Repository for #{project.name} was created"
|
|
| 37 |
repository = Repository.factory('Subversion', :project => project, :url => url)
|
|
| 38 |
repository.save |
|
| 42 |
if project && project.repository.auto |
|
| 43 |
repository = project.repository |
|
| 44 |
repository.url = url |
|
| 45 |
logger.debug "Repository for #{project.name} was created"
|
|
| 46 |
repository.save |
|
| 47 |
end |
|
| 39 | 48 |
repository.id || 0 |
| 40 | 49 |
end |
| 41 | 50 | |
| app/helpers/repositories_helper.rb | ||
|---|---|---|
| 42 | 42 |
str |
| 43 | 43 |
end |
| 44 | 44 |
|
| 45 |
def repository_field_tags(form, repository) |
|
| 46 |
method = repository.class.name.demodulize.underscore + "_field_tags" |
|
| 47 |
send(method, form, repository) if repository.is_a?(Repository) && respond_to?(method) |
|
| 48 |
end |
|
| 49 |
|
|
| 45 |
def repository_field_tags(form, repository) |
|
| 46 |
if repository |
|
| 47 |
method = repository.class.name.demodulize.underscore + "_field_tags" |
|
| 48 |
if !repository.auto || (repository.nil? || repository.new_record?) |
|
| 49 |
send(method, form, repository) if repository.is_a?(Repository) && respond_to?(method) |
|
| 50 |
end |
|
| 51 |
end |
|
| 52 | ||
| 53 |
def repository_auto(repository) |
|
| 54 |
check_box ('repository', 'auto',
|
|
| 55 |
:checked => (repository.nil? || repository.new_record?) ? false : repository.auto, |
|
| 56 |
:disabled => (repository && !repository.new_record?), |
|
| 57 |
:onclick => "if (this.checked) {Element.hide('url_fields');} else {Element.show('url_fields');}" )
|
|
| 58 |
end |
|
| 59 | ||
| 50 | 60 |
def scm_select_tag(repository) |
| 51 | 61 |
container = [[]] |
| 52 | 62 |
REDMINE_SUPPORTED_SCM.each {|scm| container << ["Repository::#{scm}".constantize.scm_name, scm]}
|
| 53 | 63 |
select_tag('repository_scm',
|
| 54 | 64 |
options_for_select(container, repository.class.name.demodulize), |
| 55 | 65 |
:disabled => (repository && !repository.new_record?), |
| 56 |
:onchange => remote_function(:url => { :controller => 'repositories', :action => 'edit', :id => @project }, :method => :get, :with => "Form.serialize(this.form)")
|
|
| 66 |
:onchange => remote_function(:url => { :controller => 'repositories', :action => 'edit', :id => @project }, :method => :get, :with => "Form.serialize(this.form)")
|
|
| 57 | 67 |
) |
| 58 | 68 |
end |
| 59 | 69 |
|
| app/models/repository/bazaar.rb | ||
|---|---|---|
| 19 | 19 | |
| 20 | 20 |
class Repository::Bazaar < Repository |
| 21 | 21 |
attr_protected :root_url |
| 22 |
validates_presence_of :url |
|
| 22 |
validates_presence_of :url, :unless => Proc.new { |repository| repository.auto }
|
|
| 23 | 23 | |
| 24 | 24 |
def scm_adapter |
| 25 | 25 |
Redmine::Scm::Adapters::BazaarAdapter |
| app/models/repository/cvs.rb | ||
|---|---|---|
| 19 | 19 |
require 'digest/sha1' |
| 20 | 20 | |
| 21 | 21 |
class Repository::Cvs < Repository |
| 22 |
validates_presence_of :url, :root_url |
|
| 22 |
validates_presence_of :url, :root_url, :unless => Proc.new { |repository| repository.auto }
|
|
| 23 | 23 | |
| 24 | 24 |
def scm_adapter |
| 25 | 25 |
Redmine::Scm::Adapters::CvsAdapter |
| app/models/repository/darcs.rb | ||
|---|---|---|
| 18 | 18 |
require 'redmine/scm/adapters/darcs_adapter' |
| 19 | 19 | |
| 20 | 20 |
class Repository::Darcs < Repository |
| 21 |
validates_presence_of :url |
|
| 21 |
validates_presence_of :url, :unless => Proc.new { |repository| repository.auto }
|
|
| 22 | 22 | |
| 23 | 23 |
def scm_adapter |
| 24 | 24 |
Redmine::Scm::Adapters::DarcsAdapter |
| app/models/repository/git.rb | ||
|---|---|---|
| 19 | 19 | |
| 20 | 20 |
class Repository::Git < Repository |
| 21 | 21 |
attr_protected :root_url |
| 22 |
validates_presence_of :url |
|
| 22 |
validates_presence_of :url, :unless => Proc.new { |repository| repository.auto }
|
|
| 23 | 23 | |
| 24 | 24 |
def scm_adapter |
| 25 | 25 |
Redmine::Scm::Adapters::GitAdapter |
| app/models/repository/mercurial.rb | ||
|---|---|---|
| 19 | 19 | |
| 20 | 20 |
class Repository::Mercurial < Repository |
| 21 | 21 |
attr_protected :root_url |
| 22 |
validates_presence_of :url |
|
| 22 |
validates_presence_of :url, :unless => Proc.new { |repository| repository.auto }
|
|
| 23 | 23 | |
| 24 | 24 |
def scm_adapter |
| 25 | 25 |
Redmine::Scm::Adapters::MercurialAdapter |
| app/models/repository/subversion.rb | ||
|---|---|---|
| 19 | 19 | |
| 20 | 20 |
class Repository::Subversion < Repository |
| 21 | 21 |
attr_protected :root_url |
| 22 |
validates_presence_of :url |
|
| 23 |
validates_format_of :url, :with => /^(http|https|svn|svn\+ssh|file):\/\/.+/i |
|
| 22 |
validates_presence_of :url, :unless => Proc.new { |repository| repository.auto }
|
|
| 23 |
validates_format_of :url, :with => /^(http|https|svn|svn\+ssh|file):\/\/.+/i, :unless => Proc.new { |repository| repository.auto }
|
|
| 24 | 24 | |
| 25 | 25 |
def scm_adapter |
| 26 | 26 |
Redmine::Scm::Adapters::SubversionAdapter |
| app/views/projects/settings/_repository.rhtml | ||
|---|---|---|
| 7 | 7 | |
| 8 | 8 |
<div class="box tabular"> |
| 9 | 9 |
<p><label><%= l(:label_scm) %></label><%= scm_select_tag(@repository) %></p> |
| 10 |
<%= repository_field_tags(f, @repository) if @repository %> |
|
| 10 |
<p><label><%= l(:label_scm_auto) %></label><%= repository_auto(@repository) %></p> |
|
| 11 |
<div id="url_fields"><%= repository_field_tags(f, @repository) %></div> |
|
| 11 | 12 |
</div> |
| 12 | 13 | |
| 13 | 14 |
<div class="contextual"> |
| db/migrate/094_add_automatic_repository_column.rb | ||
|---|---|---|
| 1 |
class AddAutomaticRepositoryColumn < ActiveRecord::Migration |
|
| 2 |
def self.up |
|
| 3 |
add_column :repositories, :auto, :boolean, :default => false, :null => false |
|
| 4 |
end |
|
| 5 | ||
| 6 |
def self.down |
|
| 7 |
remove_column :repositories, :auto |
|
| 8 |
end |
|
| 9 |
end |
|
| extra/svn/reposmangen.rb | ||
|---|---|---|
| 1 |
#!/usr/bin/ruby |
|
| 2 | ||
| 3 |
# == Synopsis |
|
| 4 |
# |
|
| 5 |
# reposman: manages your svn repositories with Redmine |
|
| 6 |
# |
|
| 7 |
# == Usage |
|
| 8 |
# |
|
| 9 |
# reposman [ -h | --help ] [ -v | --verbose ] [ -V | --version ] [ -q | --quiet ] -s /var/svn -r redmine.host.org |
|
| 10 |
# example: reposman --svn-dir=/var/svn --redmine-host=redmine.mydomain.foo |
|
| 11 |
# reposman -s /var/svn -r redmine.mydomain.foo |
|
| 12 |
# |
|
| 13 |
# == Arguments (mandatory) |
|
| 14 |
# |
|
| 15 |
# -s, --svn-dir=DIR |
|
| 16 |
# use DIR as base directory for svn repositories |
|
| 17 |
# |
|
| 18 |
# -r, --redmine-host=HOST |
|
| 19 |
# assume Redmine is hosted on HOST. |
|
| 20 |
# you can use : |
|
| 21 |
# * -r redmine.mydomain.foo (will add http://) |
|
| 22 |
# * -r http://redmine.mydomain.foo |
|
| 23 |
# * -r https://mydomain.foo/redmine |
|
| 24 |
# |
|
| 25 |
# == Options |
|
| 26 |
# |
|
| 27 |
# -o, --owner=OWNER |
|
| 28 |
# owner of the repository. using the rails login allow user to browse |
|
| 29 |
# the repository in Redmine even for private project |
|
| 30 |
# |
|
| 31 |
# -u, --url=URL |
|
| 32 |
# the base url Redmine will use to access your repositories. This |
|
| 33 |
# will be used to register the repository in Redmine so that user |
|
| 34 |
# doesn't need to do anything. reposman will add the identifier to this url : |
|
| 35 |
# |
|
| 36 |
# -u https://my.svn.server/my/reposity/root # if the repository can be access by http |
|
| 37 |
# -u file:///var/svn/ # if the repository is local |
|
| 38 |
# if this option isn't set, reposman won't register the repository |
|
| 39 |
# |
|
| 40 |
# -t, --test |
|
| 41 |
# only show what should be done |
|
| 42 |
# |
|
| 43 |
# -h, --help: |
|
| 44 |
# show help and exit |
|
| 45 |
# |
|
| 46 |
# -v, --verbose |
|
| 47 |
# verbose |
|
| 48 |
# |
|
| 49 |
# -V, --version |
|
| 50 |
# print version and exit |
|
| 51 |
# |
|
| 52 |
# -q, --quiet |
|
| 53 |
# no log |
|
| 54 |
# |
|
| 55 | ||
| 56 |
require 'getoptlong' |
|
| 57 |
require 'rdoc/usage' |
|
| 58 |
require 'soap/wsdlDriver' |
|
| 59 |
require 'find' |
|
| 60 |
require 'etc' |
|
| 61 |
require 'yaml' |
|
| 62 | ||
| 63 |
Version = "1.0" |
|
| 64 | ||
| 65 |
opts = GetoptLong.new( |
|
| 66 |
['--conf', '-c', GetoptLong::REQUIRED_ARGUMENT], |
|
| 67 |
['--redmine-host', '-r', GetoptLong::REQUIRED_ARGUMENT], |
|
| 68 |
['--owner', '-o', GetoptLong::REQUIRED_ARGUMENT], |
|
| 69 |
['--url', '-u', GetoptLong::REQUIRED_ARGUMENT], |
|
| 70 |
['--test', '-t', GetoptLong::NO_ARGUMENT], |
|
| 71 |
['--verbose', '-v', GetoptLong::NO_ARGUMENT], |
|
| 72 |
['--version', '-V', GetoptLong::NO_ARGUMENT], |
|
| 73 |
['--help' , '-h', GetoptLong::NO_ARGUMENT], |
|
| 74 |
['--quiet' , '-q', GetoptLong::NO_ARGUMENT] |
|
| 75 |
) |
|
| 76 | ||
| 77 |
$conffile = "config.yml" |
|
| 78 |
$verbose = 0 |
|
| 79 |
$quiet = false |
|
| 80 |
$redmine_host = '' |
|
| 81 |
$vcs_owner = 'root' |
|
| 82 |
$use_groupid = true |
|
| 83 |
$test = false |
|
| 84 | ||
| 85 |
def log(text,level=0, exit=false) |
|
| 86 |
return if $quiet or level > $verbose |
|
| 87 |
puts text |
|
| 88 |
exit 1 if exit |
|
| 89 |
end |
|
| 90 | ||
| 91 |
begin |
|
| 92 |
opts.each do |opt, arg| |
|
| 93 |
case opt |
|
| 94 |
when '--conf'; $conffile = arg.dup |
|
| 95 |
when '--redmine-host'; $redmine_host = arg.dup |
|
| 96 |
when '--owner'; $vcs_owner = arg.dup; $use_groupid = false; |
|
| 97 |
when '--verbose'; $verbose += 1 |
|
| 98 |
when '--test'; $test = true |
|
| 99 |
when '--version'; puts Version; exit |
|
| 100 |
when '--help'; RDoc::usage |
|
| 101 |
when '--quiet'; $quiet = true |
|
| 102 |
end |
|
| 103 |
end |
|
| 104 |
rescue |
|
| 105 |
exit 1 |
|
| 106 |
end |
|
| 107 | ||
| 108 |
if File.exist?($conffile) |
|
| 109 |
config = open($conffile) {|f| YAML.load(f) }
|
|
| 110 |
else |
|
| 111 |
config = {
|
|
| 112 |
"Git" => {"url" => "http://example.org/git",
|
|
| 113 |
"base" => "/path/to/git" } , |
|
| 114 |
"Subversion" => {
|
|
| 115 |
"url" => "http://subversion.example.org/", |
|
| 116 |
"base" => "/path/to/subversion" } |
|
| 117 |
} |
|
| 118 |
open('config.yml', 'w') {|f| YAML.dump(config, f)}
|
|
| 119 |
log("Creating stub config file. Please edit it.",-1,true)
|
|
| 120 |
end |
|
| 121 | ||
| 122 |
log("running in test mode") if $test
|
|
| 123 | ||
| 124 |
RDoc::usage if $redmine_host.empty? |
|
| 125 | ||
| 126 |
log("querying Redmine for projects...", 1);
|
|
| 127 | ||
| 128 |
$redmine_host.gsub!(/^/, "http://") unless $redmine_host.match("^https?://")
|
|
| 129 |
$redmine_host.gsub!(/\/$/, '') |
|
| 130 | ||
| 131 |
wsdl_url = "#{$redmine_host}/sys/service.wsdl";
|
|
| 132 | ||
| 133 |
begin |
|
| 134 |
soap = SOAP::WSDLDriverFactory.new(wsdl_url).create_rpc_driver |
|
| 135 |
rescue => e |
|
| 136 |
log("Unable to connect to #{wsdl_url} : #{e}", 0, true)
|
|
| 137 |
end |
|
| 138 | ||
| 139 |
projects = soap.Projects |
|
| 140 | ||
| 141 |
if projects.nil? |
|
| 142 |
log('no project found, perhaps you forgot to "Enable WS for repository management"', 0, true)
|
|
| 143 |
end |
|
| 144 | ||
| 145 |
log("retrieved #{projects.size} projects", 1)
|
|
| 146 | ||
| 147 |
def set_owner_and_rights(project, repos_path, &block) |
|
| 148 |
if RUBY_PLATFORM =~ /mswin/ |
|
| 149 |
yield if block_given? |
|
| 150 |
else |
|
| 151 |
uid = Etc.getpwnam($vcs_owner).uid |
|
| 152 |
# if the gid is 0 I get an exception... |
|
| 153 |
#gid = $use_groupid ? Etc.getgrnam(project.identifier).gid : 0 |
|
| 154 |
gid = 0 |
|
| 155 |
right = project.is_public ? 0775 : 0770 |
|
| 156 |
yield if block_given? |
|
| 157 |
Find.find(repos_path) do |f| |
|
| 158 |
File.chmod right, f |
|
| 159 |
File.chown uid, gid, f |
|
| 160 |
end |
|
| 161 |
end |
|
| 162 |
end |
|
| 163 | ||
| 164 |
def other_read_right?(file) |
|
| 165 |
(File.stat(file).mode & 0007).zero? ? false : true |
|
| 166 |
end |
|
| 167 | ||
| 168 |
def owner_name(file) |
|
| 169 |
RUBY_PLATFORM =~ /mswin/ ? |
|
| 170 |
$vcs_owner : |
|
| 171 |
Etc.getpwuid( File.stat(file).uid ).name |
|
| 172 |
end |
|
| 173 | ||
| 174 |
projects.each do |element| |
|
| 175 |
project = element.project |
|
| 176 |
repository = element.repository |
|
| 177 | ||
| 178 |
repos_type = repository["type"] |
|
| 179 |
repos_base = config["#{repos_type}"]["base"]
|
|
| 180 |
repos_url = config["#{repos_type}"]["url"]
|
|
| 181 |
repos_url += "/" if repos_url and not repos_url.match(/\/$/) |
|
| 182 |
repos_path = repos_base + "/" + project.identifier |
|
| 183 | ||
| 184 |
unless File.directory?(repos_base) |
|
| 185 |
log("directory '#{repos_base}' doesn't exists", 0, true)
|
|
| 186 |
end |
|
| 187 | ||
| 188 |
log("treating project #{project.name}", 1)
|
|
| 189 | ||
| 190 |
if project.identifier.empty? |
|
| 191 |
log("\tno identifier for project #{project.name}")
|
|
| 192 |
next |
|
| 193 |
elsif not project.identifier.match(/^[a-z0-9\-]+$/) |
|
| 194 |
log("\tinvalid identifier for project #{project.name} : #{project.identifier}");
|
|
| 195 |
next; |
|
| 196 |
end |
|
| 197 |
|
|
| 198 |
if repository.auto |
|
| 199 |
if File.directory?(repos_path) |
|
| 200 | ||
| 201 |
# we must verify that repository has the good owner and the good |
|
| 202 |
# rights before leaving |
|
| 203 |
other_read = other_read_right?(repos_path) |
|
| 204 |
owner = owner_name(repos_path) |
|
| 205 |
next if project.is_public == other_read and owner == $vcs_owner |
|
| 206 | ||
| 207 |
if $test |
|
| 208 |
log("\tchange mode on #{repos_path}")
|
|
| 209 |
next |
|
| 210 |
end |
|
| 211 | ||
| 212 |
begin |
|
| 213 |
set_owner_and_rights(project, repos_path) |
|
| 214 |
rescue Errno::EPERM => e |
|
| 215 |
log("\tunable to change mode on #{repos_path} : #{e}\n")
|
|
| 216 |
next |
|
| 217 |
end |
|
| 218 | ||
| 219 |
log("\tmode change on #{repos_path}");
|
|
| 220 | ||
| 221 |
else |
|
| 222 |
# the directory repos_path doesn't exist |
|
| 223 |
project.is_public ? File.umask(0002) : File.umask(0007) |
|
| 224 | ||
| 225 |
if $test |
|
| 226 |
log("\tcreate repository #{repos_path}")
|
|
| 227 |
log("\trepository #{repos_path} registered in Redmine with url #{repos_url}#{project.identifier}") if repos_url;
|
|
| 228 |
next |
|
| 229 |
end |
|
| 230 | ||
| 231 |
begin |
|
| 232 |
Dir.mkdir(repos_path) unless test(?d, repos_path) |
|
| 233 |
set_owner_and_rights(project, repos_path) do |
|
| 234 |
case "#{repos_type}"
|
|
| 235 |
when "Subversion": |
|
| 236 |
raise "svnadmin create #{repos_path} failed" unless system("svnadmin", "create", repos_path)
|
|
| 237 |
when "Git": |
|
| 238 |
Dir.chdir(repos_path) do |
|
| 239 |
raise "git create #{repos_path} failed" unless system("git-init-db", "--shared")
|
|
| 240 |
end |
|
| 241 |
else |
|
| 242 |
raise "unknown cvs type #{repos_type}"
|
|
| 243 |
end |
|
| 244 |
end |
|
| 245 |
rescue => e |
|
| 246 |
Dir.rmdir(repos_path) |
|
| 247 |
log("\tunable to create #{repos_path} : #{e}\n")
|
|
| 248 |
next |
|
| 249 |
end |
|
| 250 | ||
| 251 |
if repos_url |
|
| 252 |
ret = soap.RepositoryCreated project.identifier, "#{repos_url}#{project.identifier}"
|
|
| 253 |
if ret > 0 |
|
| 254 |
log("\trepository #{repos_path} registered in Redmine with url #{repos_url}#{project.identifier}");
|
|
| 255 |
else |
|
| 256 |
log("\trepository #{repos_path} not registered in Redmine. Look in your log to find why.");
|
|
| 257 |
end |
|
| 258 |
end |
|
| 259 | ||
| 260 |
log("\trepository #{repos_path} created");
|
|
| 261 |
end |
|
| 262 |
end |
|
| 263 |
end |
|
| 264 | ||
| lang/en.yml | ||
|---|---|---|
| 501 | 501 |
label_general: General |
| 502 | 502 |
label_more: More |
| 503 | 503 |
label_scm: SCM |
| 504 |
label_scm_auto: Create Repository |
|
| 504 | 505 |
label_plugins: Plugins |
| 505 | 506 |
label_ldap_authentication: LDAP authentication |
| 506 | 507 |
label_downloads_abbr: D/L |
| lib/redmine/scm/adapters/abstract_adapter.rb | ||
|---|---|---|
| 24 | 24 |
end |
| 25 | 25 |
|
| 26 | 26 |
class AbstractAdapter #:nodoc: |
| 27 |
def initialize(url, root_url=nil, login=nil, password=nil)
|
|
| 27 |
def initialize(url=nil, root_url=nil, login=nil, password=nil)
|
|
| 28 | 28 |
@url = url |
| 29 | 29 |
@login = login if login && !login.empty? |
| 30 | 30 |
@password = (password || "") if @login |