{config, pkgs, serverInfo}: let inherit (pkgs.lib) mkOption; urlPrefix = config.urlPrefix; dbDir = "${config.dataDir}/db"; reposDir = "${config.dataDir}/repos"; backupsDir = "${config.dataDir}/backup"; distsDir = "${config.dataDir}/dist"; tmpDir = "${config.dataDir}/tmp"; logDir = "${config.dataDir}/log"; postCommitHook = "/var/run/current-system/sw/bin/svn-server-post-commit-hook"; fsType = "fsfs"; adminAddr = serverInfo.serverConfig.adminAddr; # Build a Subversion instance with Apache modules and Swig/Python bindings. subversion = import ../../../nixpkgs/pkgs/applications/version-management/subversion-1.4.x { inherit (pkgs) fetchurl stdenv apr aprutil neon expat swig zlib; bdbSupport = true; httpServer = true; sslSupport = true; compressionSupport = true; pythonBindings = true; httpd = pkgs.apacheHttpd; }; # Build the maintenance scripts and commit hooks. scripts = substituteInAll { name = "svn-server-scripts"; src = pkgs.lib.cleanSource ../../../services/subversion/src/scripts; # The variables to substitute: inherit reposDir dbDir logDir distsDir backupsDir tmpDir urlPrefix adminAddr fsType subversion postCommitHook; inherit (config) notificationSender userCreationDomain; orgUrl = config.organisation.url; orgLogoUrl = config.organisation.logo; orgName = config.organisation.name; perl = "${pkgs.perl}/bin/perl"; sendmail = "${pkgs.ssmtp}/sbin/sendmail"; inherit (pkgs) libxslt enscript db4 coreutils bzip2; inherit (serverInfo) canonicalName; # Urgh, most of these are dependencies of Email::Send, should figure them out automatically. perlFlags = map (x: "-I${x}/lib/site_perl") [ pkgs.perlBerkeleyDB pkgs.perlEmailSend pkgs.perlEmailSimple pkgs.perlModulePluggable pkgs.perlReturnValue pkgs.perlEmailAddress pkgs.perlCryptPasswordMD5 pkgs.perlStringMkPasswd ]; # Do a syntax check on the generated file. postInstall = '' $perl -c -T $out/cgi-bin/repoman.pl $perl -c $out/bin/svn-server-create-user.pl ''; }; # Build our custom authentication modules. authModules = import ../../../services/subversion/src/auth { inherit (pkgs) stdenv apacheHttpd; }; commonAuth = '' AuthType Basic AuthName "Subversion repositories" AuthBasicProvider dbm AuthDBMType DB AuthDBMUserFile ${dbDir}/svn-users ''; # Access controls for /repos and /repos-xml. reposConfig = dirName: '' ${commonAuth} AuthAllowNone on AuthzRepoPrefix ${urlPrefix}/${dirName}/ AuthzRepoDBType DB AuthzRepoReaders ${dbDir}/svn-readers AuthzRepoWriters ${dbDir}/svn-writers Require repo-writer Require repo-reader DAV svn SVNParentPath ${reposDir} SVNAutoversioning ${if config.autoVersioning then "on" else "off"} ''; # Build ViewVC. viewvc = import ../../../services/subversion/src/viewvc { inherit (pkgs) fetchurl stdenv python enscript; inherit urlPrefix reposDir adminAddr subversion; }; viewerConfig = dirName: '' ${commonAuth} AuthAllowNone on AuthzRepoPrefix ${urlPrefix}/${dirName}/ AuthzRepoDBType DB AuthzRepoReaders ${dbDir}/svn-readers Require repo-reader ''; viewvcConfig = '' ScriptAlias ${urlPrefix}/viewvc ${viewvc}/viewvc/bin/mod_python/viewvc.py AddHandler python-program .py # Note: we write \" instead of ' to work around a lexer bug in Nix 0.11. PythonPath "[\"${viewvc}/viewvc/bin/mod_python\", \"${subversion}/lib/python2.4/site-packages\"] + sys.path" PythonHandler handler ${viewerConfig "viewvc"} Alias ${urlPrefix}/viewvc-doc ${viewvc}/viewvc/templates/docroot Redirect permanent ${urlPrefix}/viewcvs ${serverInfo.canonicalName}/${urlPrefix}/viewvc ''; # Build WebSVN. websvn = import ../../../services/subversion/src/websvn { inherit (pkgs) fetchurl stdenv writeText enscript gnused diffutils; inherit urlPrefix reposDir subversion; cacheDir = tmpDir; }; websvnConfig = '' Alias ${urlPrefix}/websvn ${websvn}/wsvn.php Alias ${urlPrefix}/templates ${websvn}/templates ${viewerConfig "websvn"} Order allow,deny Allow from all ''; distConfig = '' Alias ${urlPrefix}/dist ${distsDir} AllowOverride None Options Indexes FollowSymLinks Order allow,deny Allow from all IndexOptions +SuppressDescription +NameWidth=* IndexIgnore *.rev *.lock IndexStyleSheet ${urlPrefix}/style.css ${viewerConfig "dist"} ''; repomanConfig = '' ScriptAlias ${urlPrefix}/repoman ${scripts}/cgi-bin/repoman.pl ${commonAuth} Require valid-user Order deny,allow Deny from all Allow from 127.0.0.1 Allow from ${config.userCreationDomain} ${commonAuth} Require valid-user ${commonAuth} Require valid-user Order deny,allow Deny from all Allow from 127.0.0.1 Allow from ${config.userCreationDomain} ${commonAuth} Require valid-user ${viewerConfig "repoman/dump"} ''; staticFiles = substituteInSome { name = "svn-static-files"; src = pkgs.lib.cleanSource ../../../services/subversion/root; files = ["xsl/svnindex.xsl"]; inherit urlPrefix; }; staticFilesConfig = '' # !!! this breaks UserDir if urlPrefix == "" Alias ${if urlPrefix == "" then "/" else urlPrefix} ${staticFiles}/ Order allow,deny Allow from all DirectoryIndex repoman ''; # !!! should be in Nixpkgs. substituteInSome = args: pkgs.stdenvUsingSetupNew2.mkDerivation ({ buildCommand = '' ensureDir $out cp -prd $src/* $out chmod -R u+w $out for i in $files; do substituteAll $out/$i $out/$i done ''; } // args); # */ substituteInAll = args: pkgs.stdenvUsingSetupNew2.mkDerivation ({ buildCommand = '' ensureDir $out cp -prd $src/* $out chmod -R u+w $out find $out -type f -print | while read fn; do substituteAll $fn $fn done eval "$postInstall" ''; } // args); # */ in { extraModulesPre = [ # Allow anonymous access to repositories that are world-readable # without prompting for a username/password. { name = "authn_noauth"; path = "${authModules}/modules/mod_authn_noauth.so"; } # Check whether the user is allowed read or write access to a # repository. { name = "authz_dyn"; path = "${authModules}/modules/mod_authz_dyn.so"; } ]; extraModules = [ { name = "python"; path = "${pkgs.mod_python}/modules/mod_python.so"; } { name = "php5"; path = "${pkgs.php}/modules/libphp5.so"; } { name = "dav_svn"; path = "${subversion}/modules/mod_dav_svn.so"; } ]; extraConfig = '' ${reposConfig "repos"} ${reposConfig "repos-xml"} SVNIndexXSLT "${urlPrefix}/xsl/svnindex.xsl" ${viewvcConfig} ${websvnConfig} ${repomanConfig} ${distConfig} ${staticFilesConfig} ''; robotsEntries = '' User-agent: * Disallow: ${urlPrefix}/viewcvs/ Disallow: ${urlPrefix}/viewvc/ Disallow: ${urlPrefix}/websvn/ Disallow: ${urlPrefix}/repos-xml/ ''; # mod_python's own Python modules must be in the initial Python # path, they cannot be set through the PythonPath directive. globalEnvVars = [ { name = "PYTHONPATH"; value = "${pkgs.mod_python}/lib/python2.4/site-packages"; } ]; extraServerPath = [ # Needed for ViewVC. "${pkgs.diffutils}/bin" "${pkgs.gnused}/bin" ]; extraPath = [scripts]; options = { urlPrefix = mkOption { default = "/subversion"; description = " The URL prefix under which the Subversion service appears. Use the empty string to have it appear in the server root. "; }; notificationSender = mkOption { default = "svn-server@example.org"; example = "svn-server@example.org"; description = " The email address used in the Sender field of commit notification messages sent by the Subversion subservice. "; }; userCreationDomain = mkOption { default = "example.org"; example = "example.org"; description = " The domain from which user creation is allowed. A client can only create a new user account if its IP address resolves to this domain. "; }; autoVersioning = mkOption { default = false; description = " Whether you want the Subversion subservice to support auto-versioning, which enables Subversion repositories to be mounted as read/writable file systems on operating systems that support WebDAV. "; }; dataDir = mkOption { default = "/no/such/path/exists"; description = " Place to put SVN repository. "; }; organisation = { name = mkOption { default = null; description = " Name of the organization hosting the Subversion service. "; }; url = mkOption { default = null; description = " URL of the website of the organization hosting the Subversion service. "; }; logo = mkOption { default = null; description = " Logo the organization hosting the Subversion service. "; }; }; }; }