Compare commits
	
		
			241 Commits
		
	
	
		
			v1.1
			...
			feature.wi
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					9a6a6c21f5 | ||
| 
						 | 
					69d0384ec3 | ||
| 
						 | 
					af956d01dd | ||
| 
						 | 
					0002e9bd21 | ||
| 
						 | 
					6c62fbd19f | ||
| 
						 | 
					7abcc29e60 | ||
| 
						 | 
					2e1369184b | ||
| 
						 | 
					b39fe8595c | ||
| 
						 | 
					5fefc26543 | ||
| 
						 | 
					a37b222878 | ||
| 
						 | 
					c9f2575469 | ||
| 
						 | 
					5101ae5f35 | ||
| 
						 | 
					bd94d5bf68 | ||
| 
						 | 
					aa09862059 | ||
| 
						 | 
					f9629f3f7b | ||
| 
						 | 
					884f50155c | ||
| 
						 | 
					cf684c1514 | ||
| 
						 | 
					6c9b6b5309 | ||
| 
						 | 
					a7f283256a | ||
| 
						 | 
					4506e1d2b5 | ||
| 
						 | 
					70f67e6bc6 | ||
| 
						 | 
					05816cb75a | ||
| 
						 | 
					91b189b75f | ||
| 
						 | 
					623c562054 | ||
| 
						 | 
					2a55024640 | ||
| 
						 | 
					8a9f8c66e8 | ||
| 
						 | 
					81f433085a | ||
| 
						 | 
					6e59a0a526 | ||
| 
						 | 
					8397d86313 | ||
| 
						 | 
					b0ac05b608 | ||
| 
						 | 
					48c3989ae3 | ||
| 
						 | 
					b949b7e83a | ||
| 
						 | 
					ad7fad9fbe | ||
| 
						 | 
					d2db3b16d2 | ||
| 
						 | 
					ae11b1de4a | ||
| 
						 | 
					6f620e3f54 | ||
| 
						 | 
					810b753edf | ||
| 
						 | 
					958fb1b9ad | ||
| 
						 | 
					48992adefa | ||
| 
						 | 
					d95e1e13e5 | ||
| 
						 | 
					381dc5b8f7 | ||
| 
						 | 
					464c1a8ef5 | ||
| 
						 | 
					e5b10a8494 | ||
| 
						 | 
					ffc49b9ea6 | ||
| 
						 | 
					0efc14dd6f | ||
| 
						 | 
					4fb15ccb7d | ||
| 
						 | 
					234b70845c | ||
| 
						 | 
					6abd0b6faa | ||
| 
						 | 
					fef2bd15bf | ||
| 
						 | 
					74d07d8fb8 | ||
| 
						 | 
					473e9153ed | ||
| 
						 | 
					efa10c9afd | ||
| 
						 | 
					7438a2bf19 | ||
| 
						 | 
					2e0995abac | ||
| 
						 | 
					c84afd0f78 | ||
| 
						 | 
					b1276dff6c | ||
| 
						 | 
					b413b7ee89 | ||
| 
						 | 
					f19f07ec59 | ||
| 
						 | 
					83761c66c5 | ||
| 
						 | 
					d0e2977746 | ||
| 
						 | 
					1be91e5a2a | ||
| 
						 | 
					15d4d1aa7d | ||
| 
						 | 
					708b90fccd | ||
| 
						 | 
					ac7a4c4aa5 | ||
| 
						 | 
					b90246a239 | ||
| 
						 | 
					a9d327d54f | ||
| 
						 | 
					eb8fd0aa55 | ||
| 
						 | 
					813184f06c | ||
| 
						 | 
					ef2d3a9af9 | ||
| 
						 | 
					9a8bd464a3 | ||
| 
						 | 
					9e2ea7404b | ||
| 
						 | 
					160d11b89b | ||
| 
						 | 
					d860f299fd | ||
| 
						 | 
					33882d4fa7 | ||
| 
						 | 
					85978a4d18 | ||
| 
						 | 
					e1e7696d53 | ||
| 
						 | 
					695428075b | ||
| 
						 | 
					e7c2e721b4 | ||
| 
						 | 
					13fad756ab | ||
| 
						 | 
					1f0791df0e | ||
| 
						 | 
					a2c832a130 | ||
| 
						 | 
					b17de014ec | ||
| 
						 | 
					b1b72190e1 | ||
| 
						 | 
					2ff2f888bc | ||
| 
						 | 
					57c2389aae | ||
| 
						 | 
					d54c86f813 | ||
| 
						 | 
					05a9697007 | ||
| 
						 | 
					945429abf0 | ||
| 
						 | 
					a016bcb51b | ||
| 
						 | 
					f2b1ce795c | ||
| 
						 | 
					3a8c56acc4 | ||
| 
						 | 
					7b2552f940 | ||
| 
						 | 
					324b202215 | ||
| 
						 | 
					2c2da6082a | ||
| 
						 | 
					dd3fbbd7e4 | ||
| 
						 | 
					9bbcd571ec | ||
| 
						 | 
					bbc185bf3b | ||
| 
						 | 
					d1bcdcda20 | ||
| 
						 | 
					6d55602ef3 | ||
| 
						 | 
					6e7c9f7c4b | ||
| 
						 | 
					5427aab456 | ||
| 
						 | 
					4879d64989 | ||
| 
						 | 
					dab8ea63fc | ||
| 
						 | 
					b03d7a04a0 | ||
| 
						 | 
					ef5b93e3f7 | ||
| 
						 | 
					69ae1c08ef | ||
| 
						 | 
					8e4f828cc6 | ||
| 
						 | 
					c4f92f4569 | ||
| 
						 | 
					c4d2b99656 | ||
| 
						 | 
					d4fe88adab | ||
| 
						 | 
					69d0e8313a | ||
| 
						 | 
					11a234e135 | ||
| 
						 | 
					2f30e4e2f6 | ||
| 
						 | 
					118ca9f11f | ||
| 
						 | 
					24fc41ee0d | ||
| 
						 | 
					c00dbac5e0 | ||
| 
						 | 
					d7857c5126 | ||
| 
						 | 
					ac6be0d3c0 | ||
| 
						 | 
					7ff6f09f67 | ||
| 
						 | 
					00b576c5a3 | ||
| 
						 | 
					2a33510c96 | ||
| 
						 | 
					d1f79d906d | ||
| 
						 | 
					aa043f14ac | ||
| 
						 | 
					638b28399e | ||
| 
						 | 
					1d86f036a3 | ||
| 
						 | 
					2f6e0f0a22 | ||
| 
						 | 
					1b1b00a10c | ||
| 
						 | 
					8693418d39 | ||
| 
						 | 
					8ff15368ce | ||
| 
						 | 
					32cde534bd | ||
| 
						 | 
					c0e26133bd | ||
| 
						 | 
					592c2ff9ff | ||
| 
						 | 
					30efd0a2db | ||
| 
						 | 
					20c3f14cc8 | ||
| 
						 | 
					80313c88c8 | ||
| 
						 | 
					cab1c09ffc | ||
| 
						 | 
					c421919092 | ||
| 
						 | 
					d350876bc1 | ||
| 
						 | 
					de09c8af56 | ||
| 
						 | 
					998f4576fe | ||
| 
						 | 
					06c57f7da6 | ||
| 
						 | 
					4d5418a601 | ||
| 
						 | 
					95cc7f627f | ||
| 
						 | 
					2aab4eea3b | ||
| 
						 | 
					5bbff9f5a6 | ||
| 
						 | 
					b5fcf1636a | ||
| 
						 | 
					13012be5d7 | ||
| 
						 | 
					ca2ef814fb | ||
| 
						 | 
					b9407f6aee | ||
| 
						 | 
					9bcb5f9456 | ||
| 
						 | 
					f412099f69 | ||
| 
						 | 
					0aa5999bb3 | ||
| 
						 | 
					16dda0743c | ||
| 
						 | 
					d079838818 | ||
| 
						 | 
					81ce4688df | ||
| 
						 | 
					ee33cc1832 | ||
| 
						 | 
					82bb18fe10 | ||
| 
						 | 
					8987ca7db6 | ||
| 
						 | 
					8066fd8982 | ||
| 
						 | 
					a96d8c05a7 | ||
| 
						 | 
					4238a5dd50 | ||
| 
						 | 
					94da55d15e | ||
| 
						 | 
					09979b8551 | ||
| 
						 | 
					5b82efa0be | ||
| 
						 | 
					8502a36481 | ||
| 
						 | 
					bcba64b2a1 | ||
| 
						 | 
					e40d922eef | ||
| 
						 | 
					7e226b43d3 | ||
| 
						 | 
					9171bfd1ab | ||
| 
						 | 
					be4774c95c | ||
| 
						 | 
					bbf1a1882a | ||
| 
						 | 
					9644784a79 | ||
| 
						 | 
					1940d5c0b5 | ||
| 
						 | 
					7c7e3cd1f1 | ||
| 
						 | 
					3e2f95a152 | ||
| 
						 | 
					92de88ba13 | ||
| 
						 | 
					5322cdf609 | ||
| 
						 | 
					02d0f0923e | ||
| 
						 | 
					765a86ba90 | ||
| 
						 | 
					34309e34c3 | ||
| 
						 | 
					2176d1cde2 | ||
| 
						 | 
					45d53e8d21 | ||
| 
						 | 
					b36b8e3afb | ||
| 
						 | 
					801af66a4e | ||
| 
						 | 
					df6ffdf420 | ||
| 
						 | 
					b3368071ac | ||
| 
						 | 
					67b80ee11c | ||
| 
						 | 
					dc31155de1 | ||
| 
						 | 
					0bae69908b | ||
| 
						 | 
					47a077bc82 | ||
| 
						 | 
					836ff71364 | ||
| 
						 | 
					0afa07b2bd | ||
| 
						 | 
					4ee3d471fe | ||
| 
						 | 
					29c7fed81b | ||
| 
						 | 
					fb62061e5a | ||
| 
						 | 
					9b92b7139f | ||
| 
						 | 
					5ea4b02205 | ||
| 
						 | 
					12d3eef3d1 | ||
| 
						 | 
					576c06ffaf | ||
| 
						 | 
					352dc3e179 | ||
| 
						 | 
					aa164936f4 | ||
| 
						 | 
					9a93acd1a5 | ||
| 
						 | 
					587aa11cda | ||
| 
						 | 
					d04ecd60c4 | ||
| 
						 | 
					ecef510f78 | ||
| 
						 | 
					4976c20935 | ||
| 
						 | 
					5553c37ccd | ||
| 
						 | 
					346b2c6cf8 | ||
| 
						 | 
					aa68fe3485 | ||
| 
						 | 
					e408fe8733 | ||
| 
						 | 
					836986462a | ||
| 
						 | 
					51c6cdb20d | ||
| 
						 | 
					766232f29b | ||
| 
						 | 
					2ed021f30b | ||
| 
						 | 
					899fe561df | ||
| 
						 | 
					67d8936083 | ||
| 
						 | 
					78f5cef5bd | ||
| 
						 | 
					8928b1654c | ||
| 
						 | 
					977fa0d1d4 | ||
| 
						 | 
					1d89cec2cf | ||
| 
						 | 
					6c6b7d6bb2 | ||
| 
						 | 
					f92e88e9b4 | ||
| 
						 | 
					6e58899fdf | ||
| 
						 | 
					1513598420 | ||
| 
						 | 
					9cfc060760 | ||
| 
						 | 
					a531e34ec2 | ||
| 
						 | 
					2cce23b5b4 | ||
| 
						 | 
					e518bc9b68 | ||
| 
						 | 
					8e02d05ba9 | ||
| 
						 | 
					f131083315 | ||
| 
						 | 
					39ba5b37ef | ||
| 
						 | 
					002fa05c7f | ||
| 
						 | 
					1a52133fd4 | ||
| 
						 | 
					30900f7196 | ||
| 
						 | 
					b753cf0837 | ||
| 
						 | 
					53ab5b6aff | ||
| 
						 | 
					78a0402351 | ||
| 
						 | 
					5b5705fe90 | ||
| 
						 | 
					f08b5c5e3f | ||
| 
						 | 
					aa87acd432 | ||
| 
						 | 
					7af7ef8357 | 
							
								
								
									
										20
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -1,11 +1,19 @@
 | 
			
		||||
*~
 | 
			
		||||
tmp
 | 
			
		||||
.buildpath
 | 
			
		||||
.externalToolBuilders
 | 
			
		||||
.project
 | 
			
		||||
.settings
 | 
			
		||||
.tx/config
 | 
			
		||||
attachments
 | 
			
		||||
indefero-*.zip
 | 
			
		||||
src/IDF/conf/idf.php
 | 
			
		||||
src/IDF/conf/idf.test.php
 | 
			
		||||
www/test.php
 | 
			
		||||
www/media/upload
 | 
			
		||||
src/IDF/gettexttemplates
 | 
			
		||||
indefero-*.zip
 | 
			
		||||
src/IDF/conf/path.php
 | 
			
		||||
.tx/config
 | 
			
		||||
src/IDF/gettexttemplates
 | 
			
		||||
src/IDF/locale/idf.pot.bak
 | 
			
		||||
test/config.php
 | 
			
		||||
test/test.db
 | 
			
		||||
test/tmp
 | 
			
		||||
tmp
 | 
			
		||||
www/media/upload
 | 
			
		||||
www/test.php
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										8
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								AUTHORS
									
									
									
									
									
								
							@@ -12,24 +12,32 @@ Much appreciated contributors (in alphabetical order):
 | 
			
		||||
    Brian Armstrong <brianar>
 | 
			
		||||
    Charles Melbye <charlie@yourwiki.net>
 | 
			
		||||
    Ciaran Gultnieks <ciaran@ciarang.com>
 | 
			
		||||
    Daniel Steudler <steudlerdaniel@gmail.com>
 | 
			
		||||
    David Feeney <davidf>
 | 
			
		||||
    Denis Kot <denis.kot@gmail.com>                 - Russian translation
 | 
			
		||||
    Dmitry Dulepov <dmitryd>
 | 
			
		||||
    Fernando Sayago Gil <mikados.mikados@gmail.com> - Spanish translation
 | 
			
		||||
    Gert van Valkenhoef <gertvv>
 | 
			
		||||
    Jakub Viták <mainiak@gmail.com>                 - Czech translation
 | 
			
		||||
    Janez Troha <http://www.dz0ny.info>             - Slovenian translation
 | 
			
		||||
    Jean-Philippe Fleury <jpfleury>
 | 
			
		||||
    Jerry <lxb429@gmail.com>                        - Chinese translation
 | 
			
		||||
    Julien Issler <julien@issler.net>
 | 
			
		||||
    Litew <litew9@gmail.com>                        - Russian translation
 | 
			
		||||
    Ludovic Bellière <xrogaan>
 | 
			
		||||
    Manuel Eidenberger <eidenberger@gmail.com>
 | 
			
		||||
    Matthew Dawson <mjd>
 | 
			
		||||
    Matías Halles <matias@halles.cl>
 | 
			
		||||
    Mehdi Kabab <http://pioupioum.fr/>
 | 
			
		||||
    Nicolas Lassalle <nicolas@beroot.org>           - Subversion support
 | 
			
		||||
    Ozan <uobasar@gmail.com>                        - Turkish translation
 | 
			
		||||
    Patrick Georgi <patrick.georgi@coresystems.de>
 | 
			
		||||
    Pedro Kiefer <pedro@kiefer.com.br>              - Brazilian Portuguese translation
 | 
			
		||||
    Raphaël Emourgeon <raphael>
 | 
			
		||||
    Samuel Suther <info@suther.de>                  - German translation
 | 
			
		||||
    Sindre R. Myren <sindrero@stud.ntnu.no>
 | 
			
		||||
    Stewart Platt <stew@futurete.ch>
 | 
			
		||||
    Stéphane Baron <sbaron>
 | 
			
		||||
    Thomas Keller <me@thomaskeller.biz>             - Monotone support
 | 
			
		||||
    Vladimir Solomatin <slash>
 | 
			
		||||
    William Martin <william.martin@lcpc.fr>
 | 
			
		||||
 
 | 
			
		||||
@@ -4,6 +4,12 @@ The installation of InDefero is composed of 2 parts, first the
 | 
			
		||||
installation of the [Pluf framework](http://www.pluf.org) and second,
 | 
			
		||||
the installation of InDefero by itself.
 | 
			
		||||
 | 
			
		||||
## PHP modules for indefero
 | 
			
		||||
 | 
			
		||||
Indefero need the GD module for PHP. It's named "php5-gd" in debian.
 | 
			
		||||
 | 
			
		||||
    $ apt-get install php5-gd
 | 
			
		||||
 | 
			
		||||
## Recommended Layout of the Files
 | 
			
		||||
 | 
			
		||||
If your server document root is in `/var/www` a good thing is to keep
 | 
			
		||||
@@ -126,6 +132,7 @@ The documentation is available in the `doc` folder.
 | 
			
		||||
* Subversion: `doc/syncsvn.mdtext`.
 | 
			
		||||
* Mercurial: `doc/syncmercurial.mdtext`.
 | 
			
		||||
* Git: `doc/syncgit.mdtext`.
 | 
			
		||||
* Monotone: `doc/syncmonotone.mdtext`
 | 
			
		||||
 | 
			
		||||
## For the Apache Webserver Users
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										15
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								Makefile
									
									
									
									
									
								
							@@ -37,6 +37,10 @@ help:
 | 
			
		||||
	@printf "\tpo-push    - Send the all PO files to the transifex server.\n"
 | 
			
		||||
	@printf "\tpo-pull    - Get all PO files from the transifex server.\n"
 | 
			
		||||
	@printf "\tpo-stats   - Show translation statistics of all PO files.\n"
 | 
			
		||||
	@printf "\nMisc Rules :\n";
 | 
			
		||||
	@printf "\tdb-install - Install the database schema.\n"
 | 
			
		||||
	@printf "\tdb-update  - Update the database schema.\n"
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#
 | 
			
		||||
#   Internationalization rule, POT & PO file manipulation
 | 
			
		||||
@@ -58,11 +62,11 @@ pot-update: pluf_path
 | 
			
		||||
	fi
 | 
			
		||||
	touch src/IDF/locale/idf.pot;
 | 
			
		||||
	# Extract string
 | 
			
		||||
	@cd src; php $(PLUF_PATH)/extracttemplates.php IDF/conf/idf.php IDF/gettexttemplates
 | 
			
		||||
	@cd src; php "$(PLUF_PATH)/extracttemplates.php" IDF/conf/idf.php IDF/gettexttemplates
 | 
			
		||||
	@cd src; for phpfile in `find . -iname "*.php"`; do \
 | 
			
		||||
		printf "Parsing file : "$$phpfile"\n"; \
 | 
			
		||||
		xgettext -o idf.pot -p ./IDF/locale/ --from-code=UTF-8 -j \
 | 
			
		||||
			--keyword --keyword=__ --keyword=_n:1,2 -L PHP $$phpfile ; \
 | 
			
		||||
			--keyword --keyword=__ --keyword=_n:1,2 -L PHP "$$phpfile" ; \
 | 
			
		||||
		done
 | 
			
		||||
	#	Remove tmp folder
 | 
			
		||||
	rm -Rf src/IDF/gettexttemplates
 | 
			
		||||
@@ -72,7 +76,7 @@ pot-update: pluf_path
 | 
			
		||||
po-update: pluf_path
 | 
			
		||||
	@for pofile in `ls src/IDF/locale/*/idf.po`; do \
 | 
			
		||||
		printf "Updating file : "$$pofile"\n"; \
 | 
			
		||||
		msgmerge -v -U $$pofile src/IDF/locale/idf.pot; \
 | 
			
		||||
		msgmerge -v -U "$$pofile" src/IDF/locale/idf.pot; \
 | 
			
		||||
		printf "\n"; \
 | 
			
		||||
	done
 | 
			
		||||
 | 
			
		||||
@@ -139,3 +143,8 @@ po-stats:
 | 
			
		||||
		> indefero-$(@:-zipfile=)-`git log $(@:-zipfile=) -n 1 \
 | 
			
		||||
		--pretty=format:%h`.zip
 | 
			
		||||
 | 
			
		||||
db-install:
 | 
			
		||||
	@cd src && php "$(PLUF_PATH)/migrate.php" --conf=IDF/conf/idf.php -a -d -i
 | 
			
		||||
 | 
			
		||||
db-update:
 | 
			
		||||
	@cd src && php "$(PLUF_PATH)/migrate.php" --conf=IDF/conf/idf.php -a -d
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										117
									
								
								NEWS.mdtext
									
									
									
									
									
								
							
							
						
						
									
										117
									
								
								NEWS.mdtext
									
									
									
									
									
								
							@@ -1,3 +1,120 @@
 | 
			
		||||
# InDefero 1.3 - xxx xxx xx xx:xx:xx UTC 201X
 | 
			
		||||
 | 
			
		||||
## New Features
 | 
			
		||||
 | 
			
		||||
## Bugfixes
 | 
			
		||||
 | 
			
		||||
## Documentation
 | 
			
		||||
 | 
			
		||||
## Translations
 | 
			
		||||
 | 
			
		||||
# InDefero 1.2 - Sun Nov 6 23:04:00 UTC 2011
 | 
			
		||||
 | 
			
		||||
ATTENTION: You need Pluf [46b7f251](http://projects.ceondo.com/p/pluf/source/commit/46b7f251)
 | 
			
		||||
or newer to properly run this version of Indefero!
 | 
			
		||||
 | 
			
		||||
## New Features
 | 
			
		||||
 | 
			
		||||
- Indefero's issue tracker can now bi-directionally link issues with variable,
 | 
			
		||||
  configurable terms, such as "is related to", "is blocked by" or
 | 
			
		||||
  "is duplicated by" (issue 638)
 | 
			
		||||
- When you search for issues, the results can further be refined by issue state
 | 
			
		||||
  (open or closed) and label (partially implements issue 548)
 | 
			
		||||
- Source and diff views now make characters like line endings, tabs and other
 | 
			
		||||
  "invisible" control characters visible on hover and cope with long lines much
 | 
			
		||||
  better (issue 636)
 | 
			
		||||
- Mercurial source views now show parent revisions (if any) and detailed change
 | 
			
		||||
  information
 | 
			
		||||
- Subversion source views now show detailed change information (issue 622)
 | 
			
		||||
- File download URLs now contain the file name rather than the upload id;
 | 
			
		||||
  old links still work though (issues 559 and 686)
 | 
			
		||||
- monotone file and directory attributes are displayed in the tree and file view
 | 
			
		||||
  (needs a monotone with an interface version of 13.1 or newer)
 | 
			
		||||
- The context area is now kept in view when a page scrolls down several pages
 | 
			
		||||
- A summary section has been added to the issue tracker with statistics about
 | 
			
		||||
  open / closed  issues, unresolved issues grouped by tags and owners
 | 
			
		||||
- The project list and title has gathered a customizable icon for each project
 | 
			
		||||
- The download section now provides MD5 checksums for uploaded files
 | 
			
		||||
- Wiki pages now come with a designated stylesheet for printer output (issue 713)
 | 
			
		||||
 | 
			
		||||
## Bugfixes
 | 
			
		||||
 | 
			
		||||
- Git's cron job doesn't erase manually added keys anymore (issue 247)
 | 
			
		||||
- The SVN interface acts more robust if an underlying repository has been
 | 
			
		||||
  restructured (issues 364 and 721)
 | 
			
		||||
- monotone zip archive entries now all carry the revision date as mtime (issue 645)
 | 
			
		||||
- The timeline only now only displays filter options for items a user has
 | 
			
		||||
  actually access to (issue 655)
 | 
			
		||||
- The log, tags and branches parsers for Mercurial are more robust now (issue 663)
 | 
			
		||||
- Several SSH public key parsing issues have been fixed and the check for existing,
 | 
			
		||||
  uploaded keys has been improved (issue 679)
 | 
			
		||||
- Diff views now show empty context lines for git and hg again (issue 688)
 | 
			
		||||
- The SVN command line client no longer accidential tries to store the login
 | 
			
		||||
  credentials we give him as arguments for the user executing the SVN command
 | 
			
		||||
- The usher section in the forge administration no longer displays a bogus
 | 
			
		||||
  server enty in case no monotone server is configured in the connected
 | 
			
		||||
  usher instance
 | 
			
		||||
- A timeout that popped up when Usher is restarted has been fixed (issue 695)
 | 
			
		||||
- The SyncMonotone plugin now cleans up partial artifacts it created during the
 | 
			
		||||
  addition of a new project or monotone key, in case an error popped up in the
 | 
			
		||||
  middle (issue 697)
 | 
			
		||||
- Indefero now sends the MD5 checksum as HTTP header when downloading a file from the
 | 
			
		||||
  download area; additionally, a unneeded redirect has been removed (issue 716)
 | 
			
		||||
- Source links without a specific revision did not work due to a wrong regex
 | 
			
		||||
  (issue 730)
 | 
			
		||||
- Avatar URL generation use correctly the configuration (issue 732)
 | 
			
		||||
- The SyncGit plugin no longer fails to remove a non-existing post-update hook
 | 
			
		||||
  on repository creation (issue 752)
 | 
			
		||||
- When uploading a project logo, an existing uploaded file with the same name
 | 
			
		||||
  no longer leads to an error, but is simple overwritten (fixes issue 740)
 | 
			
		||||
- The error detection and reporting in the SyncMonotone plugin has been improved
 | 
			
		||||
- The branch links users of the Subversion frontend get when they enter a wrong
 | 
			
		||||
  revision are fixed; this list is now also only displayed (for any SCM) if
 | 
			
		||||
  there are actually branches available in the repository
 | 
			
		||||
- If git's author name is not encoded in an UTF-8 compatible encoding, skip the
 | 
			
		||||
  author lookup, as we have no information what the author string is actually
 | 
			
		||||
  encoded in
 | 
			
		||||
- Indefero no longer displays an empty parents paragraph in the commit view for
 | 
			
		||||
  root revisions of a git repository
 | 
			
		||||
- Indefero now only shows the tags of the closed and not the open issues in the
 | 
			
		||||
  closed issues list
 | 
			
		||||
 | 
			
		||||
## Documentation
 | 
			
		||||
 | 
			
		||||
- The documentation on the setup of the monotone plugin has been improved.
 | 
			
		||||
 | 
			
		||||
## Translations
 | 
			
		||||
 | 
			
		||||
- The Russian translation has been enabled by default (thanks for all the great
 | 
			
		||||
  work, Denis Kot and Litew!)
 | 
			
		||||
- Brazilian Portuguese translation started (thanks to Pedro Kiefer!)
 | 
			
		||||
- Turkish translation started (thanks to Ozan!)
 | 
			
		||||
 | 
			
		||||
# InDefero 1.1.2 - Thu May 26 07:42:25 2011 UTC
 | 
			
		||||
 | 
			
		||||
## Bugfixes
 | 
			
		||||
 | 
			
		||||
- Fix tags extraction from git repository (issue 675)
 | 
			
		||||
- Fix SSH validation method (issue 671)
 | 
			
		||||
- Fix malformed URL in the RSS (issue 666)
 | 
			
		||||
- Fix validateRevision call for Mercurial Scm (issue 657)
 | 
			
		||||
 | 
			
		||||
## Translations
 | 
			
		||||
 | 
			
		||||
- Missing word in French translation (issue 672)
 | 
			
		||||
- Update Spanish translation
 | 
			
		||||
 | 
			
		||||
# InDefero 1.1.1 - Mon Mar 28 15:52 2011 UTC
 | 
			
		||||
 | 
			
		||||
## Bugfixes
 | 
			
		||||
 | 
			
		||||
- Fix an incompatibility with Python 3.1 in gitserve.py (issue 554)
 | 
			
		||||
- Fix PHP error when the commit view shows a commit with changed binary files (issue 643)
 | 
			
		||||
- A migration problem prevented the preferences page being displayed properly (issues 644 and 653)
 | 
			
		||||
- Fix PHP error when trying to create Mercurial source archives (issue 648)
 | 
			
		||||
- Improve the French translation (issue 651)
 | 
			
		||||
- Registration page missed a link to password recovery that was mentioned in a form error (issue 652)
 | 
			
		||||
 | 
			
		||||
# InDefero 1.1 - Sun Mar 20 11:44 2011 UTC
 | 
			
		||||
 | 
			
		||||
## New Features
 | 
			
		||||
 
 | 
			
		||||
@@ -25,7 +25,7 @@ If you install monotone from source (<http://monotone.ca/downloads.php>),
 | 
			
		||||
please follow the `INSTALL` document which comes with the software.
 | 
			
		||||
It contains detailed instructions, including all needed dependencies.
 | 
			
		||||
 | 
			
		||||
## Choose your indefero setup
 | 
			
		||||
## Choose your indefero (IDF) setup
 | 
			
		||||
 | 
			
		||||
The monotone plugin can be used in several different ways:
 | 
			
		||||
 | 
			
		||||
@@ -112,14 +112,40 @@ The monotone plugin can be used in several different ways:
 | 
			
		||||
            ^D
 | 
			
		||||
            $ chmod 600 usher.conf
 | 
			
		||||
 | 
			
		||||
    **ATTENTION:** Do _not_ configure a default monotone server key in `usher.conf`,
 | 
			
		||||
    otherwise the individual server key that IDF creates for each project is
 | 
			
		||||
    not used and this could cause problems.
 | 
			
		||||
 | 
			
		||||
    Your indefero www user needs later write access to `usher.conf` and
 | 
			
		||||
    `projects/`. There are two ways of setting this up:
 | 
			
		||||
 | 
			
		||||
    * Make the usher user the web user, for example via Apache's `suexec`
 | 
			
		||||
    * Use acls, like this:
 | 
			
		||||
    * Make the usher user the web user, for example via Apache's `suexec`.
 | 
			
		||||
      This is however a bit clumsy.
 | 
			
		||||
    * Preferred: Use Access Control Lists (ACLs), like this:
 | 
			
		||||
 | 
			
		||||
            #
 | 
			
		||||
            # Linux
 | 
			
		||||
            #
 | 
			
		||||
            $ setfacl -m u:www:rw usher.conf
 | 
			
		||||
            $ setfacl -m d:u:www:rwx projects/
 | 
			
		||||
            $ setfacl -m d:u:usher:rwx projects/
 | 
			
		||||
            #
 | 
			
		||||
            # FreeBSD
 | 
			
		||||
            #
 | 
			
		||||
            $ setfacl -m user:www:rw::allow usher.conf
 | 
			
		||||
            $ setfacl -m user:www:rwxp:fd:allow projects/
 | 
			
		||||
            $ setfacl -m user:usher:rwxp:fd:allow projects/
 | 
			
		||||
            #
 | 
			
		||||
            # Mac OS X
 | 
			
		||||
            #
 | 
			
		||||
            chmod +a '_www allow read,write' usher.conf
 | 
			
		||||
            chmod +a '_www allow read,write,delete,add_file,add_subdirectory,file_inherit,directory_inherit' projects/
 | 
			
		||||
            chmod +a 'usher allow read,write,delete,add_file,add_subdirectory,file_inherit,directory_inherit' projects/
 | 
			
		||||
 | 
			
		||||
      In each example's last line, `usher` is the user which is executing
 | 
			
		||||
      the usher instance. **It is very important to add this line, otherwise
 | 
			
		||||
      usher won't be able to read and write into the initial file system
 | 
			
		||||
      setup IDF creates!**
 | 
			
		||||
 | 
			
		||||
5. Wrap a daemonizer around usher, for example supervise from daemontools
 | 
			
		||||
   (<http://cr.yp.to/damontools.html>):
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										173
									
								
								logo/no_logo.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										173
									
								
								logo/no_logo.svg
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,173 @@
 | 
			
		||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
 | 
			
		||||
<!-- Created with Inkscape (http://www.inkscape.org/) -->
 | 
			
		||||
 | 
			
		||||
<svg
 | 
			
		||||
   xmlns:dc="http://purl.org/dc/elements/1.1/"
 | 
			
		||||
   xmlns:cc="http://creativecommons.org/ns#"
 | 
			
		||||
   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
 | 
			
		||||
   xmlns:svg="http://www.w3.org/2000/svg"
 | 
			
		||||
   xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
 | 
			
		||||
   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
 | 
			
		||||
   sodipodi:docname="no_logo.svg"
 | 
			
		||||
   inkscape:version="0.47 r22583"
 | 
			
		||||
   id="svg2985"
 | 
			
		||||
   height="32"
 | 
			
		||||
   width="32"
 | 
			
		||||
   version="1.1"
 | 
			
		||||
   inkscape:export-filename="/Users/tommyd/Entwicklung/indefero/www/media/idf/img/no_logo.png"
 | 
			
		||||
   inkscape:export-xdpi="89.989998"
 | 
			
		||||
   inkscape:export-ydpi="89.989998">
 | 
			
		||||
  <sodipodi:namedview
 | 
			
		||||
     pagecolor="#ffffff"
 | 
			
		||||
     bordercolor="#666666"
 | 
			
		||||
     borderopacity="1"
 | 
			
		||||
     objecttolerance="10"
 | 
			
		||||
     gridtolerance="10"
 | 
			
		||||
     guidetolerance="10"
 | 
			
		||||
     inkscape:pageopacity="0"
 | 
			
		||||
     inkscape:pageshadow="2"
 | 
			
		||||
     inkscape:window-width="1440"
 | 
			
		||||
     inkscape:window-height="852"
 | 
			
		||||
     id="namedview9"
 | 
			
		||||
     showgrid="false"
 | 
			
		||||
     inkscape:zoom="16.0625"
 | 
			
		||||
     inkscape:cx="8.5507561"
 | 
			
		||||
     inkscape:cy="16.122403"
 | 
			
		||||
     inkscape:window-x="0"
 | 
			
		||||
     inkscape:window-y="0"
 | 
			
		||||
     inkscape:window-maximized="0"
 | 
			
		||||
     inkscape:current-layer="g2847"
 | 
			
		||||
     showguides="true"
 | 
			
		||||
     inkscape:guide-bbox="true">
 | 
			
		||||
    <sodipodi:guide
 | 
			
		||||
       orientation="1,0"
 | 
			
		||||
       position="16,25.836575"
 | 
			
		||||
       id="guide3752" />
 | 
			
		||||
    <sodipodi:guide
 | 
			
		||||
       orientation="0,1"
 | 
			
		||||
       position="-18.677042,16"
 | 
			
		||||
       id="guide3754" />
 | 
			
		||||
  </sodipodi:namedview>
 | 
			
		||||
  <defs
 | 
			
		||||
     id="defs2987">
 | 
			
		||||
    <inkscape:perspective
 | 
			
		||||
       sodipodi:type="inkscape:persp3d"
 | 
			
		||||
       inkscape:vp_x="0 : 16 : 1"
 | 
			
		||||
       inkscape:vp_y="0 : 1000 : 0"
 | 
			
		||||
       inkscape:vp_z="32 : 16 : 1"
 | 
			
		||||
       inkscape:persp3d-origin="16 : 10.666667 : 1"
 | 
			
		||||
       id="perspective13" />
 | 
			
		||||
    <inkscape:perspective
 | 
			
		||||
       id="perspective2863"
 | 
			
		||||
       inkscape:persp3d-origin="0.5 : 0.33333333 : 1"
 | 
			
		||||
       inkscape:vp_z="1 : 0.5 : 1"
 | 
			
		||||
       inkscape:vp_y="0 : 1000 : 0"
 | 
			
		||||
       inkscape:vp_x="0 : 0.5 : 1"
 | 
			
		||||
       sodipodi:type="inkscape:persp3d" />
 | 
			
		||||
    <inkscape:perspective
 | 
			
		||||
       id="perspective3676"
 | 
			
		||||
       inkscape:persp3d-origin="0.5 : 0.33333333 : 1"
 | 
			
		||||
       inkscape:vp_z="1 : 0.5 : 1"
 | 
			
		||||
       inkscape:vp_y="0 : 1000 : 0"
 | 
			
		||||
       inkscape:vp_x="0 : 0.5 : 1"
 | 
			
		||||
       sodipodi:type="inkscape:persp3d" />
 | 
			
		||||
    <inkscape:perspective
 | 
			
		||||
       id="perspective3717"
 | 
			
		||||
       inkscape:persp3d-origin="0.5 : 0.33333333 : 1"
 | 
			
		||||
       inkscape:vp_z="1 : 0.5 : 1"
 | 
			
		||||
       inkscape:vp_y="0 : 1000 : 0"
 | 
			
		||||
       inkscape:vp_x="0 : 0.5 : 1"
 | 
			
		||||
       sodipodi:type="inkscape:persp3d" />
 | 
			
		||||
    <filter
 | 
			
		||||
       inkscape:collect="always"
 | 
			
		||||
       id="filter3816"
 | 
			
		||||
       x="-0.14434362"
 | 
			
		||||
       width="1.2886872"
 | 
			
		||||
       y="-0.11562817"
 | 
			
		||||
       height="1.2312563">
 | 
			
		||||
      <feGaussianBlur
 | 
			
		||||
         inkscape:collect="always"
 | 
			
		||||
         stdDeviation="1.1799243"
 | 
			
		||||
         id="feGaussianBlur3818" />
 | 
			
		||||
    </filter>
 | 
			
		||||
  </defs>
 | 
			
		||||
  <metadata
 | 
			
		||||
     id="metadata2990">
 | 
			
		||||
    <rdf:RDF>
 | 
			
		||||
      <cc:Work
 | 
			
		||||
         rdf:about="">
 | 
			
		||||
        <dc:format>image/svg+xml</dc:format>
 | 
			
		||||
        <dc:type
 | 
			
		||||
           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
 | 
			
		||||
        <dc:title></dc:title>
 | 
			
		||||
      </cc:Work>
 | 
			
		||||
    </rdf:RDF>
 | 
			
		||||
  </metadata>
 | 
			
		||||
  <g
 | 
			
		||||
     inkscape:groupmode="layer"
 | 
			
		||||
     id="layer2"
 | 
			
		||||
     inkscape:label="shadow"
 | 
			
		||||
     style="display:inline"
 | 
			
		||||
     sodipodi:insensitive="true">
 | 
			
		||||
    <g
 | 
			
		||||
       transform="translate(0.44042901,0.78704792)"
 | 
			
		||||
       id="g2847-8"
 | 
			
		||||
       style="opacity:0.79710143;fill:#000000;stroke:#000000;stroke-opacity:1;filter:url(#filter3816)">
 | 
			
		||||
      <g
 | 
			
		||||
         id="g3838-0"
 | 
			
		||||
         style="fill:#000000;stroke:#000000;stroke-opacity:1" />
 | 
			
		||||
      <g
 | 
			
		||||
         id="g2401-2"
 | 
			
		||||
         transform="matrix(0.21219597,0,0,0.21219597,-70.751966,-27.73328)"
 | 
			
		||||
         style="fill:#000000;fill-opacity:1;stroke:#000000;stroke-width:2.4000001;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
 | 
			
		||||
         inkscape:export-filename="/home/loa/Projects/indefero/logo/powered-by-indefero.png"
 | 
			
		||||
         inkscape:export-xdpi="12.330909"
 | 
			
		||||
         inkscape:export-ydpi="12.330909">
 | 
			
		||||
        <path
 | 
			
		||||
           id="path2383-4"
 | 
			
		||||
           d="m 396.19089,173.14471 c -7.67621,0.80661 -14.40195,5.39406 -19.58101,10.89131 -7.23597,7.88004 -11.69742,18.07908 -13.32198,28.60362 -1.7236,11.28173 -0.25925,23.20635 5.07686,33.37271 3.78607,7.24384 9.53161,13.92339 17.29701,16.96772 3.86478,1.53937 8.98362,1.03284 11.67912,-2.41036 2.64357,-3.5671 2.69463,-8.234 2.85756,-12.48867 0.045,-7.61054 -0.54749,-15.25544 0.45618,-22.83193 0.87131,-9.50623 4.03944,-18.56751 6.71612,-27.66851 1.16242,-4.44333 2.25094,-9.02808 1.97499,-13.64988 -0.48817,-4.62476 -3.58059,-9.31042 -8.2964,-10.4067 -1.57489,-0.44882 -3.23412,-0.48948 -4.85845,-0.37931 z"
 | 
			
		||||
           style="fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:2.4000001;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0"
 | 
			
		||||
           inkscape:connector-curvature="0" />
 | 
			
		||||
        <path
 | 
			
		||||
           id="path2391-8"
 | 
			
		||||
           d="m 433.14691,149.28687 c 7.2059,2.76589 12.51512,8.93778 16.09494,15.58815 4.94991,9.48434 6.61962,20.49058 5.46486,31.07695 -1.25505,11.34342 -5.75582,22.48271 -13.54134,30.92159 -5.53192,6.01709 -12.81048,10.98198 -21.09918,11.91276 -4.13154,0.4866 -8.94486,-1.32748 -10.65734,-5.35104 -1.63027,-4.12976 -0.4717,-8.65084 0.47212,-12.80269 1.92628,-7.36287 4.47721,-14.59393 5.4687,-22.17201 1.61875,-9.40784 0.90381,-18.98034 0.67386,-28.46402 0.0272,-4.59278 0.1624,-9.30303 1.62515,-13.69592 1.66851,-4.34082 5.86829,-8.06645 10.70716,-7.90484 1.63738,-0.0259 3.25061,0.36424 4.79107,0.89107 z"
 | 
			
		||||
           style="fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:2.4000001;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0"
 | 
			
		||||
           inkscape:connector-curvature="0" />
 | 
			
		||||
      </g>
 | 
			
		||||
    </g>
 | 
			
		||||
  </g>
 | 
			
		||||
  <g
 | 
			
		||||
     inkscape:groupmode="layer"
 | 
			
		||||
     id="layer3"
 | 
			
		||||
     inkscape:label="logo"
 | 
			
		||||
     style="display:inline">
 | 
			
		||||
    <g
 | 
			
		||||
       id="layer1"
 | 
			
		||||
       transform="translate(-0.06540759,0.09444087)">
 | 
			
		||||
      <g
 | 
			
		||||
         id="g2847">
 | 
			
		||||
        <g
 | 
			
		||||
           id="g3838" />
 | 
			
		||||
        <g
 | 
			
		||||
           id="g2401"
 | 
			
		||||
           transform="matrix(0.21219597,0,0,0.21219597,-70.751966,-27.73328)"
 | 
			
		||||
           style="fill:#e6e6e6;fill-opacity:1;stroke:#a0a0a0;stroke-width:2.4000001;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
 | 
			
		||||
           inkscape:export-filename="/home/loa/Projects/indefero/logo/powered-by-indefero.png"
 | 
			
		||||
           inkscape:export-xdpi="12.330909"
 | 
			
		||||
           inkscape:export-ydpi="12.330909">
 | 
			
		||||
          <path
 | 
			
		||||
             id="path2383"
 | 
			
		||||
             d="m 396.19089,173.14471 c -7.67621,0.80661 -14.40195,5.39406 -19.58101,10.89131 -7.23597,7.88004 -11.69742,18.07908 -13.32198,28.60362 -1.7236,11.28173 -0.25925,23.20635 5.07686,33.37271 3.78607,7.24384 9.53161,13.92339 17.29701,16.96772 3.86478,1.53937 8.98362,1.03284 11.67912,-2.41036 2.64357,-3.5671 2.69463,-8.234 2.85756,-12.48867 0.045,-7.61054 -0.54749,-15.25544 0.45618,-22.83193 0.87131,-9.50623 4.03944,-18.56751 6.71612,-27.66851 1.16242,-4.44333 2.25094,-9.02808 1.97499,-13.64988 -0.48817,-4.62476 -3.58059,-9.31042 -8.2964,-10.4067 -1.57489,-0.44882 -3.23412,-0.48948 -4.85845,-0.37931 z"
 | 
			
		||||
             style="fill:#e6e6e6;fill-opacity:1;fill-rule:nonzero;stroke:#a0a0a0;stroke-width:2.4000001;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0"
 | 
			
		||||
             inkscape:connector-curvature="0" />
 | 
			
		||||
          <path
 | 
			
		||||
             id="path2391"
 | 
			
		||||
             d="m 433.14691,149.28687 c 7.2059,2.76589 12.51512,8.93778 16.09494,15.58815 4.94991,9.48434 6.61962,20.49058 5.46486,31.07695 -1.25505,11.34342 -5.75582,22.48271 -13.54134,30.92159 -5.53192,6.01709 -12.81048,10.98198 -21.09918,11.91276 -4.13154,0.4866 -8.94486,-1.32748 -10.65734,-5.35104 -1.63027,-4.12976 -0.4717,-8.65084 0.47212,-12.80269 1.92628,-7.36287 4.47721,-14.59393 5.4687,-22.17201 1.61875,-9.40784 0.90381,-18.98034 0.67386,-28.46402 0.0272,-4.59278 0.1624,-9.30303 1.62515,-13.69592 1.66851,-4.34082 5.86829,-8.06645 10.70716,-7.90484 1.63738,-0.0259 3.25061,0.36424 4.79107,0.89107 z"
 | 
			
		||||
             style="fill:#e6e6e6;fill-opacity:1;fill-rule:nonzero;stroke:#a0a0a0;stroke-width:2.4000001;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none;stroke-dashoffset:0"
 | 
			
		||||
             inkscape:connector-curvature="0" />
 | 
			
		||||
        </g>
 | 
			
		||||
      </g>
 | 
			
		||||
    </g>
 | 
			
		||||
  </g>
 | 
			
		||||
</svg>
 | 
			
		||||
| 
		 After Width: | Height: | Size: 8.4 KiB  | 
							
								
								
									
										12
									
								
								phpunit.xml
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								phpunit.xml
									
									
									
									
									
								
							@@ -12,4 +12,16 @@
 | 
			
		||||
            <directory>test/IDF/</directory>
 | 
			
		||||
        </testsuite>
 | 
			
		||||
    </testsuites>
 | 
			
		||||
 | 
			
		||||
    <filter>
 | 
			
		||||
        <whitelist>
 | 
			
		||||
            <directory suffix=".php">src/IDF</directory>
 | 
			
		||||
            <exclude>
 | 
			
		||||
                <directory suffix=".php">src/IDF/Tests</directory>
 | 
			
		||||
                <directory suffix=".php">src/IDF/conf</directory>
 | 
			
		||||
                <file>src/IDF/version.php</file>
 | 
			
		||||
            </exclude>
 | 
			
		||||
        </whitelist>
 | 
			
		||||
    </filter>
 | 
			
		||||
</phpunit>
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										14
									
								
								run-tests
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										14
									
								
								run-tests
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,14 @@
 | 
			
		||||
#!/usr/bin/env php
 | 
			
		||||
<?php
 | 
			
		||||
 | 
			
		||||
$xmlfile = dirname(__FILE__) .'/test/report.xml';
 | 
			
		||||
passthru('phpunit --coverage-clover='.$xmlfile);
 | 
			
		||||
$xml = simplexml_load_string(file_get_contents($xmlfile));
 | 
			
		||||
unlink($xmlfile);
 | 
			
		||||
printf(
 | 
			
		||||
   '>>> code coverage %s/%s (%s%%)'."\n",
 | 
			
		||||
   $xml->project->metrics['coveredstatements'],
 | 
			
		||||
   $xml->project->metrics['statements'],
 | 
			
		||||
   round(($xml->project->metrics['coveredstatements']/(float)$xml->project->metrics['statements']) * 100.0, 2)
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008-2010 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,9 @@
 | 
			
		||||
#!/usr/bin/env python
 | 
			
		||||
# -*- coding: utf-8 -*-
 | 
			
		||||
# 
 | 
			
		||||
#
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -23,14 +23,17 @@
 | 
			
		||||
 | 
			
		||||
import os
 | 
			
		||||
import sys
 | 
			
		||||
import commands
 | 
			
		||||
import traceback
 | 
			
		||||
import subprocess
 | 
			
		||||
 | 
			
		||||
SCRIPTDIR = os.path.abspath(__file__).rsplit(os.path.sep, 1)[0]
 | 
			
		||||
GITSERVEPHP = '%s/gitserve.php' % SCRIPTDIR
 | 
			
		||||
process = subprocess.Popen(['php', GITSERVEPHP, sys.argv[1]],
 | 
			
		||||
                stdout=subprocess.PIPE, stderr=subprocess.PIPE)
 | 
			
		||||
output = str.encode("\n").join(process.communicate()).strip()
 | 
			
		||||
status = process.wait()
 | 
			
		||||
 | 
			
		||||
n = len("/gitserve.py")
 | 
			
		||||
GITSERVEPHP = '%s/gitserve.php' % traceback.extract_stack(limit=1)[0][0][0:-n]
 | 
			
		||||
status, output = commands.getstatusoutput('php %s %s' % (GITSERVEPHP, sys.argv[1]))
 | 
			
		||||
if status == 0:
 | 
			
		||||
    os.execvp('git', ['git', 'shell', '-c', output.strip()])
 | 
			
		||||
else:
 | 
			
		||||
    sys.stderr.write("%s\n" % output)
 | 
			
		||||
    sys.stderr.write("%s\n" % output.strip())
 | 
			
		||||
sys.exit(1)
 | 
			
		||||
 
 | 
			
		||||
@@ -4,7 +4,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008-2010 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008-2010 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008-2010 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008-2010 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										233
									
								
								src/IDF/Diff.php
									
									
									
									
									
								
							
							
						
						
									
										233
									
								
								src/IDF/Diff.php
									
									
									
									
									
								
							@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -35,9 +35,7 @@ class IDF_Diff
 | 
			
		||||
    public function __construct($diff, $path_strip_level = 0)
 | 
			
		||||
    {
 | 
			
		||||
        $this->path_strip_level = $path_strip_level;
 | 
			
		||||
        // this works because in unified diff format even empty lines are
 | 
			
		||||
        // either prefixed with a '+', '-' or ' '
 | 
			
		||||
        $this->lines = preg_split("/\015\012|\015|\012/", $diff, -1, PREG_SPLIT_NO_EMPTY);
 | 
			
		||||
        $this->lines = IDF_FileUtil::splitIntoLines($diff, true);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function parse()
 | 
			
		||||
@@ -66,12 +64,12 @@ class IDF_Diff
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            // use new file name by default
 | 
			
		||||
            preg_match("/^\+\+\+ ([^\t]+)/", $newfileline, $m);
 | 
			
		||||
            preg_match("/^\+\+\+ ([^\t\n\r]+)/", $newfileline, $m);
 | 
			
		||||
            $current_file = $m[1];
 | 
			
		||||
            if ($current_file === '/dev/null') {
 | 
			
		||||
                // except if it's /dev/null, use the old one instead
 | 
			
		||||
                // eg. mtn 0.48 and newer
 | 
			
		||||
                preg_match("/^--- ([^\t]+)/", $oldfileline, $m);
 | 
			
		||||
                preg_match("/^--- ([^\t\r\n]+)/", $oldfileline, $m);
 | 
			
		||||
                $current_file = $m[1];
 | 
			
		||||
            }
 | 
			
		||||
            if ($this->path_strip_level > 0) {
 | 
			
		||||
@@ -101,11 +99,12 @@ class IDF_Diff
 | 
			
		||||
                $files[$current_file]['chunks'][] = array();
 | 
			
		||||
 | 
			
		||||
                while ($i < $diffsize && ($addlines >= 0 || $dellines >= 0)) {
 | 
			
		||||
                    $linetype = $this->lines[$i] != '' ? $this->lines[$i][0] : ' ';
 | 
			
		||||
                    $linetype = $this->lines[$i] != '' ? $this->lines[$i][0] : false;
 | 
			
		||||
                    $content = substr($this->lines[$i], 1);
 | 
			
		||||
                    switch ($linetype) {
 | 
			
		||||
                        case ' ':
 | 
			
		||||
                            $files[$current_file]['chunks'][$current_chunk][] =
 | 
			
		||||
                                array($delstart, $addstart, substr($this->lines[$i++], 1));
 | 
			
		||||
                                array($delstart, $addstart, $content);
 | 
			
		||||
                            $dellines--;
 | 
			
		||||
                            $addlines--;
 | 
			
		||||
                            $delstart++;
 | 
			
		||||
@@ -113,23 +112,26 @@ class IDF_Diff
 | 
			
		||||
                            break;
 | 
			
		||||
                        case '+':
 | 
			
		||||
                            $files[$current_file]['chunks'][$current_chunk][] =
 | 
			
		||||
                                array('', $addstart, substr($this->lines[$i++], 1));
 | 
			
		||||
                                array('', $addstart, $content);
 | 
			
		||||
                            $addlines--;
 | 
			
		||||
                            $addstart++;
 | 
			
		||||
                            break;
 | 
			
		||||
                        case '-':
 | 
			
		||||
                            $files[$current_file]['chunks'][$current_chunk][] =
 | 
			
		||||
                                array($delstart, '', substr($this->lines[$i++], 1));
 | 
			
		||||
                                array($delstart, '', $content);
 | 
			
		||||
                            $dellines--;
 | 
			
		||||
                            $delstart++;
 | 
			
		||||
                            break;
 | 
			
		||||
                        case '\\':
 | 
			
		||||
                            // ignore newline handling for now, see issue 636
 | 
			
		||||
                            $i++;
 | 
			
		||||
                            // no new line at the end of this file; remove pseudo new line from last line
 | 
			
		||||
                            $cur = count($files[$current_file]['chunks'][$current_chunk]) - 1;
 | 
			
		||||
                            $files[$current_file]['chunks'][$current_chunk][$cur][2] =
 | 
			
		||||
                                rtrim($files[$current_file]['chunks'][$current_chunk][$cur][2], "\r\n");
 | 
			
		||||
                            continue;
 | 
			
		||||
                        default:
 | 
			
		||||
                            break 2;
 | 
			
		||||
                    }
 | 
			
		||||
                    $i++;
 | 
			
		||||
                }
 | 
			
		||||
                $current_chunk++;
 | 
			
		||||
            }
 | 
			
		||||
@@ -144,46 +146,92 @@ class IDF_Diff
 | 
			
		||||
    public function as_html()
 | 
			
		||||
    {
 | 
			
		||||
        $out = '';
 | 
			
		||||
        foreach ($this->files as $filename=>$file) {
 | 
			
		||||
        foreach ($this->files as $filename => $file) {
 | 
			
		||||
            $pretty = '';
 | 
			
		||||
            $fileinfo = IDF_FileUtil::getMimeType($filename);
 | 
			
		||||
            if (IDF_FileUtil::isSupportedExtension($fileinfo[2])) {
 | 
			
		||||
                $pretty = ' prettyprint';
 | 
			
		||||
            }
 | 
			
		||||
            $out .= "\n".'<table class="diff" summary="">'."\n";
 | 
			
		||||
            $out .= '<tr id="diff-'.md5($filename).'"><th colspan="3">'.Pluf_esc($filename).'</th></tr>'."\n";
 | 
			
		||||
 | 
			
		||||
            $cc = 1;
 | 
			
		||||
            $offsets = array();
 | 
			
		||||
            $contents = array();
 | 
			
		||||
 | 
			
		||||
            foreach ($file['chunks'] as $chunk) {
 | 
			
		||||
                foreach ($chunk as $line) {
 | 
			
		||||
                    if ($line[0] and $line[1]) {
 | 
			
		||||
                        $class = 'diff-c';
 | 
			
		||||
                    } elseif ($line[0]) {
 | 
			
		||||
                        $class = 'diff-r';
 | 
			
		||||
                    list($left, $right, $content) = $line;
 | 
			
		||||
                    if ($left and $right) {
 | 
			
		||||
                        $class = 'context';
 | 
			
		||||
                    } elseif ($left) {
 | 
			
		||||
                        $class = 'removed';
 | 
			
		||||
                    } else {
 | 
			
		||||
                        $class = 'diff-a';
 | 
			
		||||
                        $class = 'added';
 | 
			
		||||
                    }
 | 
			
		||||
                    $line_content = self::padLine(Pluf_esc($line[2]));
 | 
			
		||||
                    $out .= sprintf('<tr class="diff-line"><td class="diff-lc">%s</td><td class="diff-lc">%s</td><td class="%s%s mono">%s</td></tr>'."\n", $line[0], $line[1], $class, $pretty, $line_content);
 | 
			
		||||
 | 
			
		||||
                    $offsets[] = sprintf('<td>%s</td><td>%s</td>', $left, $right);
 | 
			
		||||
                    $content = IDF_FileUtil::emphasizeControlCharacters(Pluf_esc($content));
 | 
			
		||||
                    $contents[] = sprintf('<td class="%s%s mono">%s</td>', $class, $pretty, $content);
 | 
			
		||||
                }
 | 
			
		||||
                if (count($file['chunks']) > $cc) {
 | 
			
		||||
                    $offsets[]  = '<td class="next">...</td><td class="next">...</td>';
 | 
			
		||||
                    $contents[] = '<td class="next"></td>';
 | 
			
		||||
                }
 | 
			
		||||
                if (count($file['chunks']) > $cc)
 | 
			
		||||
                $out .= '<tr class="diff-next"><td>...</td><td>...</td><td> </td></tr>'."\n";
 | 
			
		||||
                $cc++;
 | 
			
		||||
            }
 | 
			
		||||
            $out .= '</table>';
 | 
			
		||||
        }
 | 
			
		||||
        return Pluf_Template::markSafe($out);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public static function padLine($line)
 | 
			
		||||
    {
 | 
			
		||||
        $line = str_replace("\t", '    ', $line);
 | 
			
		||||
        $n = strlen($line);
 | 
			
		||||
        for ($i=0;$i<$n;$i++) {
 | 
			
		||||
            if (substr($line, $i, 1) != ' ') {
 | 
			
		||||
                break;
 | 
			
		||||
            list($added, $removed) = end($file['chunks_def']);
 | 
			
		||||
 | 
			
		||||
            $added = $added[0] + $added[1];
 | 
			
		||||
            $leftwidth = 0;
 | 
			
		||||
            if ($added > 0)
 | 
			
		||||
                $leftwidth = ((ceil(log10($added)) + 1) * 8) + 12;
 | 
			
		||||
 | 
			
		||||
            $removed = $removed[0] + $removed[1];
 | 
			
		||||
            $rightwidth = 0;
 | 
			
		||||
            if ($removed > 0)
 | 
			
		||||
                $rightwidth = ((ceil(log10($removed)) + 1) * 8) + 12;
 | 
			
		||||
 | 
			
		||||
            // we need to correct the width of a single column a little
 | 
			
		||||
            // to take less space and to hide the empty one
 | 
			
		||||
            $class = '';
 | 
			
		||||
            if ($leftwidth == 0) {
 | 
			
		||||
                $class = 'left-hidden';
 | 
			
		||||
                $rightwidth -= floor(log10($removed));
 | 
			
		||||
            }
 | 
			
		||||
            else if ($rightwidth == 0) {
 | 
			
		||||
                $class = 'right-hidden';
 | 
			
		||||
                $leftwidth -= floor(log10($added));
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            $inner_linecounts =
 | 
			
		||||
              '<table class="diff-linecounts '.$class.'">' ."\n".
 | 
			
		||||
                '<colgroup><col width="'.$leftwidth.'" /><col width="'. $rightwidth.'" /></colgroup>' ."\n".
 | 
			
		||||
                '<tr class="line">' .
 | 
			
		||||
                  implode('</tr>'."\n".'<tr class="line">', $offsets).
 | 
			
		||||
                '</tr>' ."\n".
 | 
			
		||||
              '</table>' ."\n";
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            $inner_contents =
 | 
			
		||||
              '<table class="diff-contents">' ."\n".
 | 
			
		||||
                '<tr class="line">' .
 | 
			
		||||
                  implode('</tr>'."\n".'<tr class="line">', $contents) .
 | 
			
		||||
                '</tr>' ."\n".
 | 
			
		||||
              '</table>' ."\n";
 | 
			
		||||
 | 
			
		||||
            $out .= '<table class="diff unified">' ."\n".
 | 
			
		||||
                      '<colgroup><col width="'.($leftwidth + $rightwidth + 1).'" /><col width="*" /></colgroup>' ."\n".
 | 
			
		||||
                      '<tr id="diff-'.md5($filename).'">'.
 | 
			
		||||
                        '<th colspan="2">'.Pluf_esc($filename).'</th>'.
 | 
			
		||||
                      '</tr>' ."\n".
 | 
			
		||||
                      '<tr>' .
 | 
			
		||||
                        '<td>'. $inner_linecounts .'</td>'. "\n".
 | 
			
		||||
                        '<td><div class="scroll">'. $inner_contents .'</div></td>'.
 | 
			
		||||
                      '</tr>' ."\n".
 | 
			
		||||
                    '</table>' ."\n";
 | 
			
		||||
        }
 | 
			
		||||
        return str_repeat(' ', $i).substr($line, $i);
 | 
			
		||||
 | 
			
		||||
        return Pluf_Template::markSafe($out);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
@@ -208,12 +256,12 @@ class IDF_Diff
 | 
			
		||||
     */
 | 
			
		||||
    public function fileCompare($orig, $chunks, $filename, $context=10)
 | 
			
		||||
    {
 | 
			
		||||
        $orig_lines = preg_split("/\015\012|\015|\012/", $orig);
 | 
			
		||||
        $orig_lines = IDF_FileUtil::splitIntoLines($orig);
 | 
			
		||||
        $new_chunks = $this->mergeChunks($orig_lines, $chunks, $context);
 | 
			
		||||
        return $this->renderCompared($new_chunks, $filename);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function mergeChunks($orig_lines, $chunks, $context=10)
 | 
			
		||||
    private function mergeChunks($orig_lines, $chunks, $context=10)
 | 
			
		||||
    {
 | 
			
		||||
        $spans = array();
 | 
			
		||||
        $new_chunks = array();
 | 
			
		||||
@@ -310,38 +358,115 @@ class IDF_Diff
 | 
			
		||||
        return $nnew_chunks;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function renderCompared($chunks, $filename)
 | 
			
		||||
    private function renderCompared($chunks, $filename)
 | 
			
		||||
    {
 | 
			
		||||
        $fileinfo = IDF_FileUtil::getMimeType($filename);
 | 
			
		||||
        $pretty = '';
 | 
			
		||||
        if (IDF_FileUtil::isSupportedExtension($fileinfo[2])) {
 | 
			
		||||
            $pretty = ' prettyprint';
 | 
			
		||||
        }
 | 
			
		||||
        $out = '';
 | 
			
		||||
 | 
			
		||||
        $cc = 1;
 | 
			
		||||
        $i = 0;
 | 
			
		||||
        $left_offsets   = array();
 | 
			
		||||
        $left_contents  = array();
 | 
			
		||||
        $right_offsets  = array();
 | 
			
		||||
        $right_contents = array();
 | 
			
		||||
 | 
			
		||||
        $max_lineno_left = $max_lineno_right = 0;
 | 
			
		||||
 | 
			
		||||
        foreach ($chunks as $chunk) {
 | 
			
		||||
            foreach ($chunk as $line) {
 | 
			
		||||
                $line1 = ' ';
 | 
			
		||||
                $line2 = ' ';
 | 
			
		||||
                $line[2] = (strlen($line[2])) ? self::padLine(Pluf_esc($line[2])) : ' ';
 | 
			
		||||
                $left    = '';
 | 
			
		||||
                $right   = '';
 | 
			
		||||
                $content = IDF_FileUtil::emphasizeControlCharacters(Pluf_esc($line[2]));
 | 
			
		||||
 | 
			
		||||
                if ($line[0] and $line[1]) {
 | 
			
		||||
                    $class = 'diff-c';
 | 
			
		||||
                    $line1 = $line2 = $line[2];
 | 
			
		||||
                    $class = 'context';
 | 
			
		||||
                    $left = $right = $content;
 | 
			
		||||
                } elseif ($line[0]) {
 | 
			
		||||
                    $class = 'diff-r';
 | 
			
		||||
                    $line1 = $line[2];
 | 
			
		||||
                    $class = 'removed';
 | 
			
		||||
                    $left = $content;
 | 
			
		||||
                } else {
 | 
			
		||||
                    $class = 'diff-a';
 | 
			
		||||
                    $line2 = $line[2];
 | 
			
		||||
                    $class = 'added';
 | 
			
		||||
                    $right = $content;
 | 
			
		||||
                }
 | 
			
		||||
                $out .= sprintf('<tr class="diff-line"><td class="diff-lc">%s</td><td class="%s mono%s"><code>%s</code></td><td class="diff-lc">%s</td><td class="%s mono%s"><code>%s</code></td></tr>'."\n", $line[0], $class, $pretty, $line1, $line[1], $class, $pretty, $line2);
 | 
			
		||||
 | 
			
		||||
                $left_offsets[]   = sprintf('<td>%s</td>', $line[0]);
 | 
			
		||||
                $right_offsets[]  = sprintf('<td>%s</td>', $line[1]);
 | 
			
		||||
                $left_contents[]  = sprintf('<td class="%s%s mono">%s</td>', $class, $pretty, $left);
 | 
			
		||||
                $right_contents[] = sprintf('<td class="%s%s mono">%s</td>', $class, $pretty, $right);
 | 
			
		||||
 | 
			
		||||
                $max_lineno_left  = max($max_lineno_left, $line[0]);
 | 
			
		||||
                $max_lineno_right = max($max_lineno_right, $line[1]);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (count($chunks) > $cc) {
 | 
			
		||||
                $left_offsets[]   = '<td class="next">...</td>';
 | 
			
		||||
                $right_offsets[]  = '<td class="next">...</td>';
 | 
			
		||||
                $left_contents[]  = '<td></td>';
 | 
			
		||||
                $right_contents[] = '<td></td>';
 | 
			
		||||
            }
 | 
			
		||||
            if (count($chunks) > $cc)
 | 
			
		||||
                $out .= '<tr class="diff-next"><td>...</td><td> </td><td>...</td><td> </td></tr>'."\n";
 | 
			
		||||
            $cc++;
 | 
			
		||||
            $i++;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $leftwidth = 1;
 | 
			
		||||
        if ($max_lineno_left > 0)
 | 
			
		||||
            $leftwidth = ((ceil(log10($max_lineno_left)) + 1) * 8) + 12;
 | 
			
		||||
 | 
			
		||||
        $rightwidth = 1;
 | 
			
		||||
        if ($max_lineno_right > 0)
 | 
			
		||||
            $rightwidth = ((ceil(log10($max_lineno_right)) + 1) * 8) + 12;
 | 
			
		||||
 | 
			
		||||
        $inner_linecounts_left =
 | 
			
		||||
          '<table class="diff-linecounts">' ."\n".
 | 
			
		||||
            '<colgroup><col width="'.$leftwidth.'" /></colgroup>' ."\n".
 | 
			
		||||
            '<tr class="line">' .
 | 
			
		||||
              implode('</tr>'."\n".'<tr class="line">', $left_offsets).
 | 
			
		||||
            '</tr>' ."\n".
 | 
			
		||||
          '</table>' ."\n";
 | 
			
		||||
 | 
			
		||||
        $inner_linecounts_right =
 | 
			
		||||
          '<table class="diff-linecounts">' ."\n".
 | 
			
		||||
            '<colgroup><col width="'.$rightwidth.'" /></colgroup>' ."\n".
 | 
			
		||||
            '<tr class="line">' .
 | 
			
		||||
              implode('</tr>'."\n".'<tr class="line">', $right_offsets).
 | 
			
		||||
            '</tr>' ."\n".
 | 
			
		||||
          '</table>' ."\n";
 | 
			
		||||
 | 
			
		||||
        $inner_contents_left =
 | 
			
		||||
          '<table class="diff-contents">' ."\n".
 | 
			
		||||
            '<tr class="line">' .
 | 
			
		||||
              implode('</tr>'."\n".'<tr class="line">', $left_contents) .
 | 
			
		||||
            '</tr>' ."\n".
 | 
			
		||||
          '</table>' ."\n";
 | 
			
		||||
 | 
			
		||||
        $inner_contents_right =
 | 
			
		||||
          '<table class="diff-contents">' ."\n".
 | 
			
		||||
            '<tr class="line">' .
 | 
			
		||||
              implode('</tr>'."\n".'<tr class="line">', $right_contents) .
 | 
			
		||||
            '</tr>' ."\n".
 | 
			
		||||
          '</table>' ."\n";
 | 
			
		||||
 | 
			
		||||
        $out =
 | 
			
		||||
          '<table class="diff context">' ."\n".
 | 
			
		||||
            '<colgroup>' .
 | 
			
		||||
              '<col width="'.($leftwidth + 1).'" /><col width="*" />' .
 | 
			
		||||
              '<col width="'.($rightwidth + 1).'" /><col width="*" />' .
 | 
			
		||||
            '</colgroup>' ."\n".
 | 
			
		||||
            '<tr id="diff-'.md5($filename).'">'.
 | 
			
		||||
              '<th colspan="4">'.Pluf_esc($filename).'</th>'.
 | 
			
		||||
            '</tr>' ."\n".
 | 
			
		||||
            '<tr>' .
 | 
			
		||||
              '<th colspan="2">'.__('Old').'</th><th colspan="2">'.__('New').'</th>' .
 | 
			
		||||
            '</tr>'.
 | 
			
		||||
            '<tr>' .
 | 
			
		||||
              '<td>'. $inner_linecounts_left .'</td>'. "\n".
 | 
			
		||||
              '<td><div class="scroll">'. $inner_contents_left .'</div></td>'. "\n".
 | 
			
		||||
              '<td>'. $inner_linecounts_right .'</td>'. "\n".
 | 
			
		||||
              '<td><div class="scroll">'. $inner_contents_right .'</div></td>'. "\n".
 | 
			
		||||
            '</tr>' ."\n".
 | 
			
		||||
            '</table>' ."\n";
 | 
			
		||||
 | 
			
		||||
        return Pluf_Template::markSafe($out);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2010 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -65,9 +65,9 @@ class IDF_FileUtil
 | 
			
		||||
        }
 | 
			
		||||
        $table = array();
 | 
			
		||||
        $i = 1;
 | 
			
		||||
        foreach (preg_split("/\015\012|\015|\012/", $content) as $line) {
 | 
			
		||||
        foreach (self::splitIntoLines($content) as $line) {
 | 
			
		||||
            $table[] = '<tr class="c-line"><td class="code-lc" id="L'.$i.'"><a href="#L'.$i.'">'.$i.'</a></td>'
 | 
			
		||||
                .'<td class="code mono'.$pretty.'">'.IDF_Diff::padLine(Pluf_esc($line)).'</td></tr>';
 | 
			
		||||
                .'<td class="code mono'.$pretty.'">'.self::emphasizeControlCharacters(Pluf_esc($line)).'</td></tr>';
 | 
			
		||||
            $i++;
 | 
			
		||||
        }
 | 
			
		||||
        return Pluf_Template::markSafe(implode("\n", $table));
 | 
			
		||||
@@ -143,6 +143,56 @@ class IDF_FileUtil
 | 
			
		||||
        return $res;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Splits a string into separate lines while retaining the individual
 | 
			
		||||
     * line ending character for every line.
 | 
			
		||||
     *
 | 
			
		||||
     * OS9 line endings are not supported.
 | 
			
		||||
     *
 | 
			
		||||
     * @param string content
 | 
			
		||||
     * @param boolean if true, skip completely empty lines
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function splitIntoLines($content, $skipEmpty = false)
 | 
			
		||||
    {
 | 
			
		||||
        $flags = PREG_SPLIT_OFFSET_CAPTURE;
 | 
			
		||||
        if ($skipEmpty) $flags |= PREG_SPLIT_NO_EMPTY;
 | 
			
		||||
        $splitted = preg_split("/\r\n|\n/", $content, -1, $flags);
 | 
			
		||||
 | 
			
		||||
        $last_off = -1;
 | 
			
		||||
        $lines = array();
 | 
			
		||||
        while (($split = array_shift($splitted)) !== null) {
 | 
			
		||||
            if ($last_off != -1) {
 | 
			
		||||
                $lines[] .= substr($content, $last_off, $split[1] - $last_off);
 | 
			
		||||
            }
 | 
			
		||||
            $last_off = $split[1];
 | 
			
		||||
        }
 | 
			
		||||
        $lines[] = substr($content, $last_off);
 | 
			
		||||
        return $lines;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * This translates most of the C0 ASCII control characters into
 | 
			
		||||
     * their visual counterparts in the 0x24## unicode plane
 | 
			
		||||
     * (http://en.wikipedia.org/wiki/C0_and_C1_control_codes).
 | 
			
		||||
     *
 | 
			
		||||
     * We could add DEL (0x7F) to this set, but unfortunately this
 | 
			
		||||
     * is not nicely mapped to 0x247F in the control plane, but 0x2421
 | 
			
		||||
     * and adding an if expression below just for this is a little bit
 | 
			
		||||
     * of a hassle. And of course, the more esoteric ones from C1 are
 | 
			
		||||
     * missing as well...
 | 
			
		||||
     *
 | 
			
		||||
     * @param string $content
 | 
			
		||||
     * @return string
 | 
			
		||||
     */
 | 
			
		||||
    public static function emphasizeControlCharacters($content)
 | 
			
		||||
    {
 | 
			
		||||
        return preg_replace(
 | 
			
		||||
            '/([\x00-\x1F])/ue',
 | 
			
		||||
            '"<span class=\"ctrl-char\" title=\"0x".bin2hex("\\1")."\">$".bin2hex("\\1")."</span>"',
 | 
			
		||||
            $content);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Find if a given mime type is a text file.
 | 
			
		||||
     * This uses the output of the self::getMimeType function.
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -31,8 +31,11 @@
 | 
			
		||||
 */
 | 
			
		||||
class IDF_Form_Admin_ProjectCreate extends Pluf_Form
 | 
			
		||||
{
 | 
			
		||||
    public $user = null;
 | 
			
		||||
    
 | 
			
		||||
    public function initFields($extra=array())
 | 
			
		||||
    {
 | 
			
		||||
        $this->user = $extra['user'];
 | 
			
		||||
        $choices = array();
 | 
			
		||||
        $options = array(
 | 
			
		||||
                         'git' => __('git'),
 | 
			
		||||
@@ -319,6 +322,7 @@ class IDF_Form_Admin_ProjectCreate extends Pluf_Form
 | 
			
		||||
                           'labels_issue_closed' => IDF_Form_IssueTrackingConf::init_closed,
 | 
			
		||||
                           'labels_issue_predefined' =>  IDF_Form_IssueTrackingConf::init_predefined,
 | 
			
		||||
                           'labels_issue_one_max' => IDF_Form_IssueTrackingConf::init_one_max,
 | 
			
		||||
                           'issue_relations' => IDF_Form_IssueTrackingConf::init_relations,
 | 
			
		||||
                           'webhook_url' => '',
 | 
			
		||||
                           'downloads_access_rights' => 'all',
 | 
			
		||||
                           'review_access_rights' => 'all',
 | 
			
		||||
@@ -335,7 +339,6 @@ class IDF_Form_Admin_ProjectCreate extends Pluf_Form
 | 
			
		||||
                $conf->setVal($prop, $tmplconf->getVal($prop, $def));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $project->created();
 | 
			
		||||
 | 
			
		||||
        if ($this->cleaned_data['template'] == '--') {
 | 
			
		||||
            IDF_Form_MembersConf::updateMemberships($project,
 | 
			
		||||
@@ -346,6 +349,8 @@ class IDF_Form_Admin_ProjectCreate extends Pluf_Form
 | 
			
		||||
                                                    $tmpl->getMembershipData('string'));
 | 
			
		||||
        }
 | 
			
		||||
        $project->membershipsUpdated();
 | 
			
		||||
        $project->created();
 | 
			
		||||
        
 | 
			
		||||
        return $project;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,19 +3,19 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2010 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# Plume Framework is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU Lesser General Public License as published by
 | 
			
		||||
# the Free Software Foundation; either version 2.1 of the License, or
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
# the Free Software Foundation; either version 2 of the License, or
 | 
			
		||||
# (at your option) any later version.
 | 
			
		||||
#
 | 
			
		||||
# Plume Framework is distributed in the hope that it will be useful,
 | 
			
		||||
# InDefero is distributed in the hope that it will be useful,
 | 
			
		||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
			
		||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
			
		||||
# GNU Lesser General Public License for more details.
 | 
			
		||||
# GNU General Public License for more details.
 | 
			
		||||
#
 | 
			
		||||
# You should have received a copy of the GNU Lesser General Public License
 | 
			
		||||
# You should have received a copy of the GNU General Public License
 | 
			
		||||
# along with this program; if not, write to the Free Software
 | 
			
		||||
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | 
			
		||||
#
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -36,6 +36,7 @@ class IDF_Form_IssueCreate extends Pluf_Form
 | 
			
		||||
    public $user = null;
 | 
			
		||||
    public $project = null;
 | 
			
		||||
    public $show_full = false;
 | 
			
		||||
    public $relation_types = null;
 | 
			
		||||
 | 
			
		||||
    public function initFields($extra=array())
 | 
			
		||||
    {
 | 
			
		||||
@@ -45,9 +46,12 @@ class IDF_Form_IssueCreate extends Pluf_Form
 | 
			
		||||
            or $this->user->hasPerm('IDF.project-member', $this->project)) {
 | 
			
		||||
            $this->show_full = true;
 | 
			
		||||
        }
 | 
			
		||||
        $this->relation_types = $this->project->getRelationsFromConfig();
 | 
			
		||||
 | 
			
		||||
        $contentTemplate = $this->project->getConf()->getVal(
 | 
			
		||||
            'labels_issue_template', IDF_Form_IssueTrackingConf::init_template
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        $this->fields['summary'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                                      array('required' => true,
 | 
			
		||||
                                            'label' => __('Summary'),
 | 
			
		||||
@@ -76,11 +80,11 @@ class IDF_Form_IssueCreate extends Pluf_Form
 | 
			
		||||
        // case of someone allowing the upload path to be accessible
 | 
			
		||||
        // to everybody.
 | 
			
		||||
        for ($i=1;$i<4;$i++) {
 | 
			
		||||
            $filename = substr($md5, 0, 2).'/'.substr($md5, 2, 2).'/'.substr($md5, 4).'/%s.dummy'; 
 | 
			
		||||
            $filename = substr($md5, 0, 2).'/'.substr($md5, 2, 2).'/'.substr($md5, 4).'/%s.dummy';
 | 
			
		||||
            $this->fields['attachment'.$i] = new Pluf_Form_Field_File(
 | 
			
		||||
                array('required' => false,
 | 
			
		||||
                      'label' => __('Attach a file'),
 | 
			
		||||
                      'move_function_params' => 
 | 
			
		||||
                      'move_function_params' =>
 | 
			
		||||
                      array('upload_path' => $upload_path,
 | 
			
		||||
                            'upload_path_create' => true,
 | 
			
		||||
                            'file_name' => $filename,
 | 
			
		||||
@@ -109,6 +113,20 @@ class IDF_Form_IssueCreate extends Pluf_Form
 | 
			
		||||
                                                                    ),
 | 
			
		||||
                                            ));
 | 
			
		||||
 | 
			
		||||
            $this->fields['relation_type0'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                          array('required' => false,
 | 
			
		||||
                                'label' => __('This issue'),
 | 
			
		||||
                                'initial' => current($this->relation_types),
 | 
			
		||||
                                'widget_attrs' => array('size' => 15),
 | 
			
		||||
                                ));
 | 
			
		||||
 | 
			
		||||
            $this->fields['relation_issue0'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                          array('required' => false,
 | 
			
		||||
                                'label' => null,
 | 
			
		||||
                                'initial' => '',
 | 
			
		||||
                                'widget_attrs' => array('size' => 10),
 | 
			
		||||
                                ));
 | 
			
		||||
 | 
			
		||||
            /*
 | 
			
		||||
             * get predefined tags for issues from current project
 | 
			
		||||
             *
 | 
			
		||||
@@ -181,7 +199,7 @@ class IDF_Form_IssueCreate extends Pluf_Form
 | 
			
		||||
            $this->cleaned_data['label'.$i] = trim($this->cleaned_data['label'.$i]);
 | 
			
		||||
            if (strpos($this->cleaned_data['label'.$i], ':') !== false) {
 | 
			
		||||
                list($class, $name) = explode(':', $this->cleaned_data['label'.$i], 2);
 | 
			
		||||
                list($class, $name) = array(mb_strtolower(trim($class)), 
 | 
			
		||||
                list($class, $name) = array(mb_strtolower(trim($class)),
 | 
			
		||||
                                            trim($name));
 | 
			
		||||
            } else {
 | 
			
		||||
                $class = 'other';
 | 
			
		||||
@@ -215,10 +233,10 @@ class IDF_Form_IssueCreate extends Pluf_Form
 | 
			
		||||
    function clean_status()
 | 
			
		||||
    {
 | 
			
		||||
        // Check that the status is in the list of official status
 | 
			
		||||
        $tags = $this->project->getTagsFromConfig('labels_issue_open', 
 | 
			
		||||
        $tags = $this->project->getTagsFromConfig('labels_issue_open',
 | 
			
		||||
                                          IDF_Form_IssueTrackingConf::init_open,
 | 
			
		||||
                                          'Status');
 | 
			
		||||
        $tags = array_merge($this->project->getTagsFromConfig('labels_issue_closed', 
 | 
			
		||||
        $tags = array_merge($this->project->getTagsFromConfig('labels_issue_closed',
 | 
			
		||||
                                          IDF_Form_IssueTrackingConf::init_closed,
 | 
			
		||||
                                          'Status')
 | 
			
		||||
                            , $tags);
 | 
			
		||||
@@ -235,6 +253,63 @@ class IDF_Form_IssueCreate extends Pluf_Form
 | 
			
		||||
        return $this->cleaned_data['status'];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // this method is not called from Pluf_Form directly, but shared for
 | 
			
		||||
    // among all similar fields
 | 
			
		||||
    function clean_relation_type($value)
 | 
			
		||||
    {
 | 
			
		||||
        $relation_type = trim($value);
 | 
			
		||||
        if (empty($relation_type))
 | 
			
		||||
            return '';
 | 
			
		||||
 | 
			
		||||
        $found = false;
 | 
			
		||||
        foreach ($this->relation_types as $type) {
 | 
			
		||||
            if ($type == $relation_type) {
 | 
			
		||||
                $found = true;
 | 
			
		||||
                break;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        if (!$found) {
 | 
			
		||||
            throw new Pluf_Form_Invalid(__('You provided an invalid relation type.'));
 | 
			
		||||
        }
 | 
			
		||||
        return $relation_type;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    function clean_relation_type0()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->clean_relation_type($this->cleaned_data['relation_type0']);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // this method is not called from Pluf_Form directly, but shared for
 | 
			
		||||
    // among all similar fields
 | 
			
		||||
    function clean_relation_issue($value)
 | 
			
		||||
    {
 | 
			
		||||
        $issues = trim($value);
 | 
			
		||||
        if (empty($issues))
 | 
			
		||||
            return '';
 | 
			
		||||
 | 
			
		||||
        $issue_ids = preg_split('/\s*,\s*/', $issues, -1, PREG_SPLIT_NO_EMPTY);
 | 
			
		||||
        foreach ($issue_ids as $issue_id) {
 | 
			
		||||
            if (!ctype_digit($issue_id) || (int)$issue_id < 1) {
 | 
			
		||||
                throw new Pluf_Form_Invalid(sprintf(
 | 
			
		||||
                    __('The value "%s" is not a valid issue id.'), $issue_id
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
            $issue = new IDF_Issue($issue_id);
 | 
			
		||||
            if ($issue->id != $issue_id || $issue->project != $this->project->id) {
 | 
			
		||||
                throw new Pluf_Form_Invalid(sprintf(
 | 
			
		||||
                    __('The issue "%s" does not exist.'), $issue_id
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return implode(', ', $issue_ids);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    function clean_relation_issue0()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->clean_relation_issue($this->cleaned_data['relation_issue0']);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Clean the attachments post failure.
 | 
			
		||||
     */
 | 
			
		||||
@@ -298,6 +373,30 @@ class IDF_Form_IssueCreate extends Pluf_Form
 | 
			
		||||
        foreach ($tags as $tag) {
 | 
			
		||||
            $issue->setAssoc($tag);
 | 
			
		||||
        }
 | 
			
		||||
        // add relations (if any)
 | 
			
		||||
        if (!empty($this->cleaned_data['relation_type0'])) {
 | 
			
		||||
            $verb = $this->cleaned_data['relation_type0'];
 | 
			
		||||
            $other_verb = $this->relation_types[$verb];
 | 
			
		||||
            $related_issues = preg_split('/\s*,\s*/', $this->cleaned_data['relation_issue0'], -1, PREG_SPLIT_NO_EMPTY);
 | 
			
		||||
 | 
			
		||||
            foreach ($related_issues as $related_issue_id) {
 | 
			
		||||
                $related_issue = new IDF_Issue($related_issue_id);
 | 
			
		||||
                $rel = new IDF_IssueRelation();
 | 
			
		||||
                $rel->issue = $issue;
 | 
			
		||||
                $rel->verb = $verb;
 | 
			
		||||
                $rel->other_issue = $related_issue;
 | 
			
		||||
                $rel->submitter = $this->user;
 | 
			
		||||
                $rel->create();
 | 
			
		||||
 | 
			
		||||
                $other_rel = new IDF_IssueRelation();
 | 
			
		||||
                $other_rel->issue = $related_issue;
 | 
			
		||||
                $other_rel->verb = $other_verb;
 | 
			
		||||
                $other_rel->other_issue = $issue;
 | 
			
		||||
                $other_rel->submitter = $this->user;
 | 
			
		||||
                $other_rel->create();
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // add the first comment
 | 
			
		||||
        $comment = new IDF_IssueComment();
 | 
			
		||||
        $comment->issue = $issue;
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -72,12 +72,29 @@ Performance          = Performance issue
 | 
			
		||||
Usability            = Affects program usability
 | 
			
		||||
Maintainability      = Hinders future changes';
 | 
			
		||||
    const init_one_max = 'Type, Priority, Milestone';
 | 
			
		||||
    // ATTENTION: if you change something here, change the values below as well!
 | 
			
		||||
    const init_relations = 'is related to
 | 
			
		||||
blocks, is blocked by
 | 
			
		||||
duplicates, is duplicated by';
 | 
			
		||||
 | 
			
		||||
    // These are actually all noop's, but we have no other chance to
 | 
			
		||||
    // tell IDF's translation mechanism to mark the strings as translatable
 | 
			
		||||
    // FIXME: IDF should get a internal translation system for strings like
 | 
			
		||||
    // that, that can also be easily expanded by users
 | 
			
		||||
    private function noop()
 | 
			
		||||
    {
 | 
			
		||||
        __('is related to');
 | 
			
		||||
        __('blocks');
 | 
			
		||||
        __('is blocked by');
 | 
			
		||||
        __('duplicates');
 | 
			
		||||
        __('is duplicated by');
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function initFields($extra=array())
 | 
			
		||||
    {
 | 
			
		||||
        $this->fields['labels_issue_template'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                                      array('required' => false,
 | 
			
		||||
                                            'label' => __('Define an issue template to hint to the reporter to provide certain information'),
 | 
			
		||||
                                            'label' => __('Define an issue template to hint the reporter to provide certain information'),
 | 
			
		||||
                                            'initial' => self::init_template,
 | 
			
		||||
                                            'widget_attrs' => array('rows' => 7,
 | 
			
		||||
                                                                    'cols' => 75),
 | 
			
		||||
@@ -113,11 +130,20 @@ Maintainability      = Hinders future changes';
 | 
			
		||||
 | 
			
		||||
        $this->fields['labels_issue_one_max'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                                      array('required' => false,
 | 
			
		||||
                                            'label' => __('Each issue may have at most one label with each of these classes'),
 | 
			
		||||
                                            'initial' => self::init_one_max, 
 | 
			
		||||
                                            'label' => __('Each issue may have at most one label with each of these classes.'),
 | 
			
		||||
                                            'initial' => self::init_one_max,
 | 
			
		||||
                                            'widget_attrs' => array('size' => 60),
 | 
			
		||||
                                            ));
 | 
			
		||||
 | 
			
		||||
        $this->fields['issue_relations'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                                      array('required' => true,
 | 
			
		||||
                                            'label' => __('Issue relations'),
 | 
			
		||||
                                            'initial' => self::init_relations,
 | 
			
		||||
                                            'help_text' => __('You can define bidirectional relations like "is related to" or "blocks, is blocked by". For standard relations pre-configured translations exist, new relations should however be defined in a language that is understood by all project members.'),
 | 
			
		||||
                                            'widget_attrs' => array('rows' => 7,
 | 
			
		||||
                                                                    'cols' => 75),
 | 
			
		||||
                                            'widget' => 'Pluf_Form_Widget_TextareaInput',
 | 
			
		||||
                                            ));
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -39,6 +39,7 @@ class IDF_Form_IssueUpdate  extends IDF_Form_IssueCreate
 | 
			
		||||
            or $this->user->hasPerm('IDF.project-member', $this->project)) {
 | 
			
		||||
            $this->show_full = true;
 | 
			
		||||
        }
 | 
			
		||||
        $this->relation_types = $this->project->getRelationsFromConfig();
 | 
			
		||||
        if ($this->show_full) {
 | 
			
		||||
            $this->fields['summary'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                                      array('required' => true,
 | 
			
		||||
@@ -69,11 +70,11 @@ class IDF_Form_IssueUpdate  extends IDF_Form_IssueCreate
 | 
			
		||||
        // case of someone allowing the upload path to be accessible
 | 
			
		||||
        // to everybody.
 | 
			
		||||
        for ($i=1;$i<4;$i++) {
 | 
			
		||||
            $filename = substr($md5, 0, 2).'/'.substr($md5, 2, 2).'/'.substr($md5, 4).'/%s.dummy'; 
 | 
			
		||||
            $filename = substr($md5, 0, 2).'/'.substr($md5, 2, 2).'/'.substr($md5, 4).'/%s.dummy';
 | 
			
		||||
            $this->fields['attachment'.$i] = new Pluf_Form_Field_File(
 | 
			
		||||
                array('required' => false,
 | 
			
		||||
                      'label' => __('Attach a file'),
 | 
			
		||||
                      'move_function_params' => 
 | 
			
		||||
                      'move_function_params' =>
 | 
			
		||||
                      array('upload_path' => $upload_path,
 | 
			
		||||
                            'upload_path_create' => true,
 | 
			
		||||
                            'file_name' => $filename,
 | 
			
		||||
@@ -102,6 +103,52 @@ class IDF_Form_IssueUpdate  extends IDF_Form_IssueCreate
 | 
			
		||||
                                                       'size' => 15,
 | 
			
		||||
                                                                    ),
 | 
			
		||||
                                            ));
 | 
			
		||||
 | 
			
		||||
            $idx = 0;
 | 
			
		||||
            // note: clean_relation_type0 and clean_relation_issue0 already
 | 
			
		||||
            //       exist in the base class
 | 
			
		||||
            $this->fields['relation_type'.$idx] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                          array('required' => false,
 | 
			
		||||
                                'label' => __('This issue'),
 | 
			
		||||
                                'initial' => current($this->relation_types),
 | 
			
		||||
                                'widget_attrs' => array('size' => 15),
 | 
			
		||||
                                ));
 | 
			
		||||
 | 
			
		||||
            $this->fields['relation_issue'.$idx] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                          array('required' => false,
 | 
			
		||||
                                'label' => null,
 | 
			
		||||
                                'initial' => '',
 | 
			
		||||
                                'widget_attrs' => array('size' => 10),
 | 
			
		||||
                                ));
 | 
			
		||||
 | 
			
		||||
            ++$idx;
 | 
			
		||||
            $relatedIssues = $this->issue->getGroupedRelatedIssues(array(), true);
 | 
			
		||||
            foreach ($relatedIssues as $verb => $ids) {
 | 
			
		||||
                $this->fields['relation_type'.$idx] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                          array('required' => false,
 | 
			
		||||
                                'label' => __('This issue'),
 | 
			
		||||
                                'initial' => $verb,
 | 
			
		||||
                                'widget_attrs' => array('size' => 15),
 | 
			
		||||
                                ));
 | 
			
		||||
                $m = 'clean_relation_type'.$idx;
 | 
			
		||||
                $this->$m = create_function('$form', '
 | 
			
		||||
                    return $form->clean_relation_type($form->cleaned_data["relation_type'.$idx.'"]);
 | 
			
		||||
                ');
 | 
			
		||||
 | 
			
		||||
                $this->fields['relation_issue'.$idx] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                              array('required' => false,
 | 
			
		||||
                                    'label' => null,
 | 
			
		||||
                                    'initial' => implode(', ', $ids),
 | 
			
		||||
                                    'widget_attrs' => array('size' => 10),
 | 
			
		||||
                                    ));
 | 
			
		||||
                $m = 'clean_relation_issue'.$idx;
 | 
			
		||||
                $this->$m = create_function('$form', '
 | 
			
		||||
                    return $form->clean_relation_issue($form->cleaned_data["relation_issue'.$idx.'"]);
 | 
			
		||||
                ');
 | 
			
		||||
 | 
			
		||||
                ++$idx;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            $tags = $this->issue->get_tags_list();
 | 
			
		||||
            for ($i=1;$i<7;$i++) {
 | 
			
		||||
                $initial = '';
 | 
			
		||||
@@ -155,6 +202,51 @@ class IDF_Form_IssueUpdate  extends IDF_Form_IssueCreate
 | 
			
		||||
    public function clean()
 | 
			
		||||
    {
 | 
			
		||||
        $this->cleaned_data = parent::clean();
 | 
			
		||||
 | 
			
		||||
        // normalize the user's input by removing dublettes and by combining
 | 
			
		||||
        // ids from identical verbs in different input fields into one array
 | 
			
		||||
        $normRelatedIssues = array();
 | 
			
		||||
        for ($idx = 0; isset($this->cleaned_data['relation_type'.$idx]); ++$idx) {
 | 
			
		||||
            $verb = $this->cleaned_data['relation_type'.$idx];
 | 
			
		||||
            if (empty($verb))
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
            $ids = preg_split('/\s*,\s*/', $this->cleaned_data['relation_issue'.$idx],
 | 
			
		||||
                              -1, PREG_SPLIT_NO_EMPTY);
 | 
			
		||||
            if (count($ids) == 0)
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
            if (!array_key_exists($verb, $normRelatedIssues))
 | 
			
		||||
                $normRelatedIssues[$verb] = array();
 | 
			
		||||
            foreach ($ids as $id) {
 | 
			
		||||
                if (!in_array($id, $normRelatedIssues[$verb]))
 | 
			
		||||
                    $normRelatedIssues[$verb][] = $id;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // now look at any added / removed ids
 | 
			
		||||
        $added = $removed = array();
 | 
			
		||||
        $relatedIssues = $this->issue->getGroupedRelatedIssues(array(), true);
 | 
			
		||||
        $added = array_diff_key($normRelatedIssues, $relatedIssues);
 | 
			
		||||
        $removed = array_diff_key($relatedIssues, $normRelatedIssues);
 | 
			
		||||
 | 
			
		||||
        $keysToLookAt = array_keys(
 | 
			
		||||
            array_intersect_key($relatedIssues, $normRelatedIssues)
 | 
			
		||||
        );
 | 
			
		||||
        foreach ($keysToLookAt as $key) {
 | 
			
		||||
            $a = array_diff($normRelatedIssues[$key], $relatedIssues[$key]);
 | 
			
		||||
            if (count($a) > 0)
 | 
			
		||||
                $added[$key] = $a;
 | 
			
		||||
            $r = array_diff($relatedIssues[$key], $normRelatedIssues[$key]);
 | 
			
		||||
            if (count($r) > 0)
 | 
			
		||||
                $removed[$key] = $r;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // cache the added / removed data, so we do not have to
 | 
			
		||||
        // calculate that again
 | 
			
		||||
        $this->cleaned_data['_added_issue_relations'] = $added;
 | 
			
		||||
        $this->cleaned_data['_removed_issue_relations'] = $removed;
 | 
			
		||||
 | 
			
		||||
        // As soon as we know that at least one change was done, we
 | 
			
		||||
        // return the cleaned data and do not go further.
 | 
			
		||||
        if (strlen(trim($this->cleaned_data['content']))) {
 | 
			
		||||
@@ -214,6 +306,11 @@ class IDF_Form_IssueUpdate  extends IDF_Form_IssueCreate
 | 
			
		||||
                    return $this->cleaned_data;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (count($this->cleaned_data['_added_issue_relations']) != 0 ||
 | 
			
		||||
                count($this->cleaned_data['_removed_issue_relations']) != 0) {
 | 
			
		||||
                return $this->cleaned_data;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        // no changes!
 | 
			
		||||
        throw new Pluf_Form_Invalid(__('No changes were entered.'));
 | 
			
		||||
@@ -255,20 +352,22 @@ class IDF_Form_IssueUpdate  extends IDF_Form_IssueCreate
 | 
			
		||||
            foreach ($tags as $tag) {
 | 
			
		||||
                if (!Pluf_Model_InArray($tag, $oldtags)) {
 | 
			
		||||
                    if (!isset($changes['lb'])) $changes['lb'] = array();
 | 
			
		||||
                    if (!isset($changes['lb']['add'])) $changes['lb']['add'] = array();
 | 
			
		||||
                    if ($tag->class != 'Other') {
 | 
			
		||||
                        $changes['lb'][] = (string) $tag; //new tag
 | 
			
		||||
                        $changes['lb']['add'][] = (string) $tag; //new tag
 | 
			
		||||
                    } else {
 | 
			
		||||
                        $changes['lb'][] = (string) $tag->name;
 | 
			
		||||
                        $changes['lb']['add'][] = (string) $tag->name;
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            foreach ($oldtags as $tag) {
 | 
			
		||||
                if (!Pluf_Model_InArray($tag, $tags)) {
 | 
			
		||||
                    if (!isset($changes['lb'])) $changes['lb'] = array();
 | 
			
		||||
                    if (!isset($changes['lb']['rem'])) $changes['lb']['rem'] = array();
 | 
			
		||||
                    if ($tag->class != 'Other') {
 | 
			
		||||
                        $changes['lb'][] = '-'.(string) $tag; //new tag
 | 
			
		||||
                        $changes['lb']['rem'][] = (string) $tag; //new tag
 | 
			
		||||
                    } else {
 | 
			
		||||
                        $changes['lb'][] = '-'.(string) $tag->name;
 | 
			
		||||
                        $changes['lb']['rem'][] = (string) $tag->name;
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -286,6 +385,47 @@ class IDF_Form_IssueUpdate  extends IDF_Form_IssueCreate
 | 
			
		||||
                or ((!is_null($owner) and !is_null($this->issue->get_owner())) and $owner->id != $this->issue->get_owner()->id)) {
 | 
			
		||||
                $changes['ow'] = (is_null($owner)) ? '---' : $owner->login;
 | 
			
		||||
            }
 | 
			
		||||
            // Issue relations - additions
 | 
			
		||||
            foreach ($this->cleaned_data['_added_issue_relations'] as $verb => $ids) {
 | 
			
		||||
                $other_verb = $this->relation_types[$verb];
 | 
			
		||||
                foreach ($ids as $id) {
 | 
			
		||||
                    $related_issue = new IDF_Issue($id);
 | 
			
		||||
                    $rel = new IDF_IssueRelation();
 | 
			
		||||
                    $rel->issue = $this->issue;
 | 
			
		||||
                    $rel->verb = $verb;
 | 
			
		||||
                    $rel->other_issue = $related_issue;
 | 
			
		||||
                    $rel->submitter = $this->user;
 | 
			
		||||
                    $rel->create();
 | 
			
		||||
 | 
			
		||||
                    $other_rel = new IDF_IssueRelation();
 | 
			
		||||
                    $other_rel->issue = $related_issue;
 | 
			
		||||
                    $other_rel->verb = $other_verb;
 | 
			
		||||
                    $other_rel->other_issue = $this->issue;
 | 
			
		||||
                    $other_rel->submitter = $this->user;
 | 
			
		||||
                    $other_rel->create();
 | 
			
		||||
                }
 | 
			
		||||
                if (!isset($changes['rel'])) $changes['rel'] = array();
 | 
			
		||||
                if (!isset($changes['rel']['add'])) $changes['rel']['add'] = array();
 | 
			
		||||
                $changes['rel']['add'][] = $verb.' '.implode(', ', $ids);
 | 
			
		||||
            }
 | 
			
		||||
            // Issue relations - removals
 | 
			
		||||
            foreach ($this->cleaned_data['_removed_issue_relations'] as $verb => $ids) {
 | 
			
		||||
                foreach ($ids as $id) {
 | 
			
		||||
                    $db = &Pluf::db();
 | 
			
		||||
                    $table = Pluf::factory('IDF_IssueRelation')->getSqlTable();
 | 
			
		||||
                    $sql = new Pluf_SQL('verb=%s AND (
 | 
			
		||||
                                        (issue=%s AND other_issue=%s) OR
 | 
			
		||||
                                        (other_issue=%s AND issue=%s))',
 | 
			
		||||
                                        array($verb,
 | 
			
		||||
                                              $this->issue->id, $id,
 | 
			
		||||
                                              $this->issue->id, $id));
 | 
			
		||||
                    $db->execute('DELETE FROM '.$table.' WHERE '.$sql->gen());
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                if (!isset($changes['rel'])) $changes['rel'] = array();
 | 
			
		||||
                if (!isset($changes['rel']['rem'])) $changes['rel']['rem'] = array();
 | 
			
		||||
                $changes['rel']['rem'][] = $verb.' '.implode(', ', $ids);
 | 
			
		||||
            }
 | 
			
		||||
            // Update the issue
 | 
			
		||||
            $this->issue->batchAssoc('IDF_Tag', $tagids);
 | 
			
		||||
            $this->issue->summary = trim($this->cleaned_data['summary']);
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										141
									
								
								src/IDF/Form/ProjectConf.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								src/IDF/Form/ProjectConf.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,141 @@
 | 
			
		||||
<?php
 | 
			
		||||
/* -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright(C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
# the Free Software Foundation; either version 2 of the License, or
 | 
			
		||||
#(at your option) any later version.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is distributed in the hope that it will be useful,
 | 
			
		||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
			
		||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
			
		||||
# GNU General Public License for more details.
 | 
			
		||||
#
 | 
			
		||||
# You should have received a copy of the GNU General Public License
 | 
			
		||||
# along with this program; if not, write to the Free Software
 | 
			
		||||
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | 
			
		||||
#
 | 
			
		||||
# ***** END LICENSE BLOCK ***** */
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Configuration of the project.
 | 
			
		||||
 */
 | 
			
		||||
class IDF_Form_ProjectConf extends Pluf_Form
 | 
			
		||||
{
 | 
			
		||||
    public $project = null;
 | 
			
		||||
 | 
			
		||||
    public function initFields($extra=array())
 | 
			
		||||
    {
 | 
			
		||||
        $this->project = $extra['project'];
 | 
			
		||||
 | 
			
		||||
        // Basic part
 | 
			
		||||
        $this->fields['name'] = new Pluf_Form_Field_Varchar(array('required' => true,
 | 
			
		||||
                                                                  'label' => __('Name'),
 | 
			
		||||
                                                                  'initial' => $this->project->name,
 | 
			
		||||
                                                                 ));
 | 
			
		||||
        $this->fields['shortdesc'] = new Pluf_Form_Field_Varchar(array('required' => true,
 | 
			
		||||
                                                                       'label' => __('Short Description'),
 | 
			
		||||
                                                                       'initial' => $this->project->shortdesc,
 | 
			
		||||
                                                                       'widget_attrs' => array('size' => '68'),
 | 
			
		||||
                                                                      ));
 | 
			
		||||
        $this->fields['description'] = new Pluf_Form_Field_Varchar(array('required' => true,
 | 
			
		||||
                                                                         'label' => __('Description'),
 | 
			
		||||
                                                                         'initial' => $this->project->description,
 | 
			
		||||
                                                                         'widget_attrs' => array('cols' => 68,
 | 
			
		||||
                                                                                                 'rows' => 26,
 | 
			
		||||
                                                                                                ),
 | 
			
		||||
                                                                         'widget' => 'Pluf_Form_Widget_TextareaInput',
 | 
			
		||||
                                                                        ));
 | 
			
		||||
 | 
			
		||||
        // Logo part
 | 
			
		||||
        $upload_path = Pluf::f('upload_path', false);
 | 
			
		||||
        if (false === $upload_path) {
 | 
			
		||||
            throw new Pluf_Exception_SettingError(__('The "upload_path" configuration variable was not set.'));
 | 
			
		||||
        }
 | 
			
		||||
        $upload_path .= '/' . $this->project->shortname;
 | 
			
		||||
        $filename = '/%s';
 | 
			
		||||
        $this->fields['logo'] = new Pluf_Form_Field_File(array('required' => false,
 | 
			
		||||
                                                         'label' => __('Update the logo'),
 | 
			
		||||
                                                         'initial' => '',
 | 
			
		||||
                                                         'help_text' => __('The logo must be a picture with a size of 32 by 32.'),
 | 
			
		||||
                                                         'max_size' => Pluf::f('max_upload_size', 5 * 1024),
 | 
			
		||||
                                                         'move_function_params' =>
 | 
			
		||||
                                                         array('upload_path' => $upload_path,
 | 
			
		||||
                                                               'upload_path_create' => true,
 | 
			
		||||
                                                               'upload_overwrite' => true,
 | 
			
		||||
                                                               'file_name' => $filename,
 | 
			
		||||
                                                               )
 | 
			
		||||
                                                         ));
 | 
			
		||||
 | 
			
		||||
        $this->fields['logo_remove'] = new Pluf_Form_Field_Boolean(array('required' => false,
 | 
			
		||||
                                                                         'label' => __('Remove the current logo'),
 | 
			
		||||
                                                                         'initial' => false,
 | 
			
		||||
                                                                         'widget' => 'Pluf_Form_Widget_CheckboxInput',
 | 
			
		||||
                                                                         ));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * If we have uploaded a file, but the form failed remove it.
 | 
			
		||||
     *
 | 
			
		||||
     */
 | 
			
		||||
    function failed()
 | 
			
		||||
    {
 | 
			
		||||
        if (!empty($this->cleaned_data['logo'])
 | 
			
		||||
            && file_exists(Pluf::f('upload_path').'/'.$this->cleaned_data['logo'])) {
 | 
			
		||||
            unlink(Pluf::f('upload_path').'/'.$this->cleaned_data['logo']);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function clean()
 | 
			
		||||
    {
 | 
			
		||||
        if (!isset($this->cleaned_data['logo_remove'])) {
 | 
			
		||||
            $this->cleaned_data['logo_remove'] = false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return $this->cleaned_data;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function clean_logo()
 | 
			
		||||
    {
 | 
			
		||||
        if (empty($this->cleaned_data['logo'])) {
 | 
			
		||||
            return '';
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $meta = getimagesize(Pluf::f('upload_path') . '/' . $this->project->shortname . $this->cleaned_data['logo']);
 | 
			
		||||
 | 
			
		||||
        if ($meta === false) {
 | 
			
		||||
            throw new Pluf_Form_Invalid(__('Could not determine the size of the uploaded picture.'));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if ($meta[0] !== 32 || $meta[1] !== 32) {
 | 
			
		||||
            throw new Pluf_Form_Invalid(__('The picture must have a size of 32 by 32.'));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return $this->cleaned_data['logo'];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function save($commit=true)
 | 
			
		||||
    {
 | 
			
		||||
        $conf = $this->project->getConf();
 | 
			
		||||
 | 
			
		||||
        // Basic part
 | 
			
		||||
        $this->project->name = $this->cleaned_data['name'];
 | 
			
		||||
        $this->project->shortdesc = $this->cleaned_data['shortdesc'];
 | 
			
		||||
        $this->project->description = $this->cleaned_data['description'];
 | 
			
		||||
        $this->project->update();
 | 
			
		||||
 | 
			
		||||
        // Logo part
 | 
			
		||||
        if ($this->cleaned_data['logo'] !== "") {
 | 
			
		||||
            $conf->setVal('logo', $this->cleaned_data['logo']);
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->cleaned_data['logo_remove'] === true) {
 | 
			
		||||
            @unlink(Pluf::f('upload_path') . '/' . $this->project->shortname . $conf->getVal('logo'));
 | 
			
		||||
            $conf->delVal('logo');
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -52,7 +52,7 @@ class IDF_Form_ReviewFileComment extends Pluf_Form
 | 
			
		||||
                                            ));
 | 
			
		||||
        }
 | 
			
		||||
        $this->fields['content'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                                      array('required' => true,
 | 
			
		||||
                                      array('required' => false,
 | 
			
		||||
                                            'label' => __('General comment'),
 | 
			
		||||
                                            'initial' => '',
 | 
			
		||||
                                            'widget' => 'Pluf_Form_Widget_TextareaInput',
 | 
			
		||||
@@ -96,24 +96,40 @@ class IDF_Form_ReviewFileComment extends Pluf_Form
 | 
			
		||||
     */
 | 
			
		||||
    public function clean()
 | 
			
		||||
    {
 | 
			
		||||
        foreach ($this->files as $filename => $def) {
 | 
			
		||||
            if (!empty($this->cleaned_data[md5($filename)])) {
 | 
			
		||||
                return $this->cleaned_data;
 | 
			
		||||
        $isOk = false;
 | 
			
		||||
        
 | 
			
		||||
        foreach($this->files as $filename => $def) {
 | 
			
		||||
            $this->cleaned_data[md5($filename)] = trim($this->cleaned_data[md5($filename)]);
 | 
			
		||||
            if(!empty($this->cleaned_data[md5($filename)])) {
 | 
			
		||||
                $isOk = true;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        throw new Pluf_Form_Invalid(__('You need to provide comments on at least one file.'));
 | 
			
		||||
        
 | 
			
		||||
        if(!empty($this->cleaned_data['content'])) {
 | 
			
		||||
            $isOk = true;
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        if (!$isOk) {
 | 
			
		||||
            throw new Pluf_Form_Invalid(__('You need to provide your general comment about the proposal, or comments on at least one file.'));
 | 
			
		||||
        }
 | 
			
		||||
        
 | 
			
		||||
        return $this->cleaned_data;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    function clean_content()
 | 
			
		||||
    {
 | 
			
		||||
        $content = trim($this->cleaned_data['content']);
 | 
			
		||||
        if (!$this->show_full and strlen($content) == 0) {
 | 
			
		||||
            throw new Pluf_Form_Invalid(__('You need to provide your general comment about the proposal.'));
 | 
			
		||||
        if(empty($content)) {
 | 
			
		||||
            if ($this->fields['status']->initial != $this->fields['status']->value) {
 | 
			
		||||
                return __('The status have been updated.');
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            return $content;
 | 
			
		||||
        }
 | 
			
		||||
        return $content;
 | 
			
		||||
        
 | 
			
		||||
        throw new Pluf_Form_Invalid(__('This field is required.'));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Save the model in the database.
 | 
			
		||||
     *
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -317,8 +317,15 @@ class IDF_Form_UserAccount  extends Pluf_Form
 | 
			
		||||
            return '';
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (preg_match('#^ssh\-[a-z]{3}\s\S+(\s\S+)?$#', $key)) {
 | 
			
		||||
            $key = str_replace(array("\n", "\r"), '', $key);
 | 
			
		||||
        $keysearch = '';
 | 
			
		||||
        if (preg_match('#^(ssh\-(?:dss|rsa)\s+\S+)(.*)#', $key, $m)) {
 | 
			
		||||
            $basekey = preg_replace('/\s+/', ' ', $m[1]);
 | 
			
		||||
            $comment = trim(preg_replace('/[\r\n]/', ' ', $m[2]));
 | 
			
		||||
           
 | 
			
		||||
            $keysearch = $basekey.'%';
 | 
			
		||||
            $key = $basekey;
 | 
			
		||||
            if (!empty($comment))
 | 
			
		||||
                $key .= ' '.$comment;
 | 
			
		||||
 | 
			
		||||
            if (Pluf::f('idf_strong_key_check', false)) {
 | 
			
		||||
 | 
			
		||||
@@ -337,7 +344,9 @@ class IDF_Form_UserAccount  extends Pluf_Form
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else if (preg_match('#^\[pubkey [^\]]+\]\s*\S+\s*\[end\]$#', $key)) {
 | 
			
		||||
        else if (preg_match('#^\[pubkey [^\]]+\]\s*(\S+)\s*\[end\]$#', $key, $m)) {
 | 
			
		||||
            $keysearch = '%'.$m[1].'%';
 | 
			
		||||
            
 | 
			
		||||
            if (Pluf::f('idf_strong_key_check', false)) {
 | 
			
		||||
 | 
			
		||||
                // if monotone can read it, it should be valid
 | 
			
		||||
@@ -367,7 +376,7 @@ class IDF_Form_UserAccount  extends Pluf_Form
 | 
			
		||||
        if ($user) {
 | 
			
		||||
            $ruser = Pluf::factory('Pluf_User', $user);
 | 
			
		||||
            if ($ruser->id > 0) {
 | 
			
		||||
                $sql = new Pluf_SQL('content=%s', array($key));
 | 
			
		||||
                $sql = new Pluf_SQL('content LIKE %s AND user=%s', array($keysearch, $ruser->id));
 | 
			
		||||
                $keys = Pluf::factory('IDF_Key')->getList(array('filter' => $sql->gen()));
 | 
			
		||||
                if (count($keys) > 0) {
 | 
			
		||||
                    throw new Pluf_Form_Invalid(
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -41,6 +41,7 @@ Phase:Support        = Plans for user support and advocacy
 | 
			
		||||
Deprecated           = Most users should NOT reference this';
 | 
			
		||||
 | 
			
		||||
    const init_one_max = '';
 | 
			
		||||
    const wiki_default_page = '';
 | 
			
		||||
 | 
			
		||||
    public function initFields($extra=array())
 | 
			
		||||
    {
 | 
			
		||||
@@ -59,8 +60,30 @@ Deprecated           = Most users should NOT reference this';
 | 
			
		||||
                                            'initial' => self::init_one_max, 
 | 
			
		||||
                                            'widget_attrs' => array('size' => 60),
 | 
			
		||||
                                            ));
 | 
			
		||||
                                            
 | 
			
		||||
        $this->fields['wiki_default_page'] = new Pluf_Form_Field_Varchar(
 | 
			
		||||
                                      array('required' => false,
 | 
			
		||||
                                            'label' => __('Set a default wiki page instead of the page listing'),
 | 
			
		||||
                                            'initial' => self::wiki_default_page, 
 | 
			
		||||
                                            'widget_attrs' => array('size' => 60),
 | 
			
		||||
                                            ));
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    public function clean_wiki_default_page()
 | 
			
		||||
    {
 | 
			
		||||
        $pageName = trim($this->cleaned_data['wiki_default_page']);
 | 
			
		||||
        if (empty($pageName)) {
 | 
			
		||||
            return '';
 | 
			
		||||
        }
 | 
			
		||||
        $sql = new Pluf_SQL('project=%s AND title=%s', array($this->data['projectId'], $pageName));
 | 
			
		||||
        $pages = Pluf::factory('IDF_WikiPage')->getList(array('filter'=>$sql->gen()));
 | 
			
		||||
        
 | 
			
		||||
        if ($pages->count() != 1) {
 | 
			
		||||
            return '';
 | 
			
		||||
        }
 | 
			
		||||
        return $pageName;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -169,6 +169,24 @@ class IDF_Issue extends Pluf_Model
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    function getGroupedRelatedIssues($opts = array(), $idsOnly = false)
 | 
			
		||||
    {
 | 
			
		||||
        $rels = $this->get_related_issues_list(array_merge($opts, array(
 | 
			
		||||
               'view' => 'with_other_issue',
 | 
			
		||||
        )));
 | 
			
		||||
 | 
			
		||||
        $res = array();
 | 
			
		||||
        foreach ($rels as $rel) {
 | 
			
		||||
            $verb = $rel->verb;
 | 
			
		||||
            if (!array_key_exists($verb, $res)) {
 | 
			
		||||
                $res[$verb] = array();
 | 
			
		||||
            }
 | 
			
		||||
            $res[$verb][] = $idsOnly ? $rel->other_issue : $rel;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return $res;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Returns an HTML fragment used to display this issue in the
 | 
			
		||||
     * timeline.
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -155,10 +155,19 @@ class IDF_IssueComment extends Pluf_Model
 | 
			
		||||
                    $out .= __('Owner:'); break;
 | 
			
		||||
                case 'lb':
 | 
			
		||||
                    $out .= __('Labels:'); break;
 | 
			
		||||
                case 'rel':
 | 
			
		||||
                    $out .= __('Relations:'); break;
 | 
			
		||||
                }
 | 
			
		||||
                $out .= '</strong> ';
 | 
			
		||||
                if ($w == 'lb') {
 | 
			
		||||
                    $out .= Pluf_esc(implode(', ', $v));
 | 
			
		||||
                if ($w == 'lb' || $w == 'rel') {
 | 
			
		||||
                    foreach ($v as $t => $ls) {
 | 
			
		||||
                        foreach ($ls as $l) {
 | 
			
		||||
                            if ($t == 'rem') $out .= '<s>';
 | 
			
		||||
                            $out .= Pluf_esc($l);
 | 
			
		||||
                            if ($t == 'rem') $out .= '</s>';
 | 
			
		||||
                            $out .= ' ';
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                } else {
 | 
			
		||||
                    $out .= Pluf_esc($v);
 | 
			
		||||
                }
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										100
									
								
								src/IDF/IssueRelation.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								src/IDF/IssueRelation.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,100 @@
 | 
			
		||||
<?php
 | 
			
		||||
/* -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
# the Free Software Foundation; either version 2 of the License, or
 | 
			
		||||
# (at your option) any later version.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is distributed in the hope that it will be useful,
 | 
			
		||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
			
		||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
			
		||||
# GNU General Public License for more details.
 | 
			
		||||
#
 | 
			
		||||
# You should have received a copy of the GNU General Public License
 | 
			
		||||
# along with this program; if not, write to the Free Software
 | 
			
		||||
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | 
			
		||||
#
 | 
			
		||||
# ***** END LICENSE BLOCK ***** */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A relation of one issue to another
 | 
			
		||||
 */
 | 
			
		||||
class IDF_IssueRelation extends Pluf_Model
 | 
			
		||||
{
 | 
			
		||||
    public $_model = __CLASS__;
 | 
			
		||||
 | 
			
		||||
    function init()
 | 
			
		||||
    {
 | 
			
		||||
        $this->_a['table'] = 'idf_issuerelations';
 | 
			
		||||
        $this->_a['model'] = __CLASS__;
 | 
			
		||||
        $this->_a['cols'] = array(
 | 
			
		||||
                             // It is mandatory to have an "id" column.
 | 
			
		||||
                            'id' =>
 | 
			
		||||
                            array(
 | 
			
		||||
                                  'type' => 'Pluf_DB_Field_Sequence',
 | 
			
		||||
                                  'blank' => true,
 | 
			
		||||
                                  ),
 | 
			
		||||
                            'issue' =>
 | 
			
		||||
                            array(
 | 
			
		||||
                                  'type' => 'Pluf_DB_Field_Foreignkey',
 | 
			
		||||
                                  'model' => 'IDF_Issue',
 | 
			
		||||
                                  'blank' => false,
 | 
			
		||||
                                  'verbose' => __('issue'),
 | 
			
		||||
                                  'relate_name' => 'related_issues',
 | 
			
		||||
                                  ),
 | 
			
		||||
                            'verb' =>
 | 
			
		||||
                            array(
 | 
			
		||||
                                  'type' => 'Pluf_DB_Field_Text',
 | 
			
		||||
                                  'blank' => false,
 | 
			
		||||
                                  'verbose' => __('verb'),
 | 
			
		||||
                                  ),
 | 
			
		||||
                            'other_issue' =>
 | 
			
		||||
                            array(
 | 
			
		||||
                                  'type' => 'Pluf_DB_Field_Foreignkey',
 | 
			
		||||
                                  'model' => 'IDF_Issue',
 | 
			
		||||
                                  'blank' => false,
 | 
			
		||||
                                  'verbose' => __('other issue'),
 | 
			
		||||
                                  'relate_name' => 'related_other_issues',
 | 
			
		||||
                                  ),
 | 
			
		||||
                            'submitter' =>
 | 
			
		||||
                            array(
 | 
			
		||||
                                  'type' => 'Pluf_DB_Field_Foreignkey',
 | 
			
		||||
                                  'model' => 'Pluf_User',
 | 
			
		||||
                                  'blank' => false,
 | 
			
		||||
                                  'verbose' => __('submitter'),
 | 
			
		||||
                                   ),
 | 
			
		||||
                            'creation_dtime' =>
 | 
			
		||||
                            array(
 | 
			
		||||
                                  'type' => 'Pluf_DB_Field_Datetime',
 | 
			
		||||
                                  'blank' => true,
 | 
			
		||||
                                  'verbose' => __('creation date'),
 | 
			
		||||
                                  ),
 | 
			
		||||
                            );
 | 
			
		||||
        $this->_a['idx'] = array(
 | 
			
		||||
                            'creation_dtime_idx' =>
 | 
			
		||||
                            array(
 | 
			
		||||
                                  'col' => 'creation_dtime',
 | 
			
		||||
                                  'type' => 'normal',
 | 
			
		||||
                                  ),
 | 
			
		||||
                            );
 | 
			
		||||
        $issuetbl = $this->_con->pfx.'idf_issues';
 | 
			
		||||
        $this->_a['views'] = array(
 | 
			
		||||
            'with_other_issue' => array(
 | 
			
		||||
                'join' => 'INNER JOIN '.$issuetbl.' ON other_issue='.$issuetbl.'.id',
 | 
			
		||||
                'select' => $this->getSelect().', summary',
 | 
			
		||||
                'props' => array('summary' => 'other_summary'),
 | 
			
		||||
        ));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    function preSave($create=false)
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->id == '') {
 | 
			
		||||
            $this->creation_dtime = gmdate('Y-m-d H:i:s');
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -80,7 +80,10 @@ class IDF_Key extends Pluf_Model
 | 
			
		||||
        if (preg_match('#^\[pubkey ([^\]]+)\]\s*(\S+)\s*\[end\]$#', $this->content, $m)) {
 | 
			
		||||
            return array('mtn', $m[1], $m[2]);
 | 
			
		||||
        }
 | 
			
		||||
        else if (preg_match('#^ssh\-[a-z]{3}\s(\S+)(?:\s(\S+))?$#', $this->content, $m)) {
 | 
			
		||||
        else if (preg_match('#^ssh\-(?:dss|rsa)\s(\S+)(?:\s(.*))?$#', $this->content, $m)) {
 | 
			
		||||
            if (!isset($m[2])) {
 | 
			
		||||
                $m[2] = "";
 | 
			
		||||
            }
 | 
			
		||||
            return array('ssh', $m[2], $m[1]);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -88,6 +88,7 @@ class IDF_Middleware
 | 
			
		||||
                              'showuser' => 'IDF_Template_ShowUser',
 | 
			
		||||
                              'ashowuser' => 'IDF_Template_AssignShowUser',
 | 
			
		||||
                              'appversion' => 'IDF_Template_AppVersion',
 | 
			
		||||
                              'upload' => 'IDF_Template_Tag_UploadUrl',
 | 
			
		||||
                                            ));
 | 
			
		||||
        $params['modifiers'] = array_merge($params['modifiers'],
 | 
			
		||||
                                           array(
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										90
									
								
								src/IDF/Migrations/17AddIssueRelations.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								src/IDF/Migrations/17AddIssueRelations.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,90 @@
 | 
			
		||||
<?php
 | 
			
		||||
/* -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
# the Free Software Foundation; either version 2 of the License, or
 | 
			
		||||
# (at your option) any later version.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is distributed in the hope that it will be useful,
 | 
			
		||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
			
		||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
			
		||||
# GNU General Public License for more details.
 | 
			
		||||
#
 | 
			
		||||
# You should have received a copy of the GNU General Public License
 | 
			
		||||
# along with this program; if not, write to the Free Software
 | 
			
		||||
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | 
			
		||||
#
 | 
			
		||||
# ***** END LICENSE BLOCK ***** */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Add the new IDF_IssueRelation model.
 | 
			
		||||
 *
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
function IDF_Migrations_17AddIssueRelations_up($params=null)
 | 
			
		||||
{
 | 
			
		||||
    $db = Pluf::db();
 | 
			
		||||
    $schema = new Pluf_DB_Schema($db);
 | 
			
		||||
    $schema->model = new IDF_IssueRelation();
 | 
			
		||||
    $schema->createTables();
 | 
			
		||||
 | 
			
		||||
    // change the serialization format for added / removed labels in IDF_IssueComment
 | 
			
		||||
    $comments = Pluf::factory('IDF_IssueComment')->getList();
 | 
			
		||||
    foreach ($comments as $comment) {
 | 
			
		||||
        if (!isset($comment->changes['lb'])) continue;
 | 
			
		||||
        $changes = $comment->changes;
 | 
			
		||||
        $adds = $removals = array();
 | 
			
		||||
        foreach ($comment->changes['lb'] as $lb) {
 | 
			
		||||
            if (substr($lb, 0, 1) == '-')
 | 
			
		||||
                $removals[] = substr($lb, 1);
 | 
			
		||||
            else
 | 
			
		||||
                $adds[] = $lb;
 | 
			
		||||
        }
 | 
			
		||||
        $changes['lb'] = array();
 | 
			
		||||
        if (count($adds) > 0)
 | 
			
		||||
            $changes['lb']['add'] = $adds;
 | 
			
		||||
        if (count($removals) > 0)
 | 
			
		||||
            $changes['lb']['rem'] = $removals;
 | 
			
		||||
        $comment->changes = $changes;
 | 
			
		||||
        $comment->update();
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function IDF_Migrations_17AddIssueRelations_down($params=null)
 | 
			
		||||
{
 | 
			
		||||
    $db = Pluf::db();
 | 
			
		||||
    $schema = new Pluf_DB_Schema($db);
 | 
			
		||||
    $schema->model = new IDF_IssueRelation();
 | 
			
		||||
    $schema->dropTables();
 | 
			
		||||
 | 
			
		||||
    // change the serialization format for added / removed labels in IDF_IssueComment
 | 
			
		||||
    $comments = Pluf::factory('IDF_IssueComment')->getList();
 | 
			
		||||
    foreach ($comments as $comment) {
 | 
			
		||||
        $changes = $comment->changes;
 | 
			
		||||
        if (empty($changes))
 | 
			
		||||
            continue;
 | 
			
		||||
        if (isset($changes['lb'])) {
 | 
			
		||||
            $labels = array();
 | 
			
		||||
            foreach ($changes['lb'] as $type => $lbs) {
 | 
			
		||||
                if (!is_array($lbs)) {
 | 
			
		||||
                    $labels[] = $lbs;
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                foreach ($lbs as $lb) {
 | 
			
		||||
                    $labels[] = ($type == 'rem' ? '-' : '') . $lb;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            $changes['lb'] = $labels;
 | 
			
		||||
        }
 | 
			
		||||
        // while we're at it, remove any 'rel' changes
 | 
			
		||||
        unset($changes['rel']);
 | 
			
		||||
        $comment->changes = $changes;
 | 
			
		||||
        $comment->update();
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										63
									
								
								src/IDF/Migrations/18DownloadMD5.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								src/IDF/Migrations/18DownloadMD5.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,63 @@
 | 
			
		||||
<?php
 | 
			
		||||
/* -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
# the Free Software Foundation; either version 2 of the License, or
 | 
			
		||||
# (at your option) any later version.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is distributed in the hope that it will be useful,
 | 
			
		||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
			
		||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
			
		||||
# GNU General Public License for more details.
 | 
			
		||||
#
 | 
			
		||||
# You should have received a copy of the GNU General Public License
 | 
			
		||||
# along with this program; if not, write to the Free Software
 | 
			
		||||
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | 
			
		||||
#
 | 
			
		||||
# ***** END LICENSE BLOCK ***** */
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Add the md5 column for the download model.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
function IDF_Migrations_18DownloadMD5_up($params=null)
 | 
			
		||||
{
 | 
			
		||||
    // Add the row
 | 
			
		||||
    $table = Pluf::factory('IDF_Upload')->getSqlTable();
 | 
			
		||||
    $sql = array();
 | 
			
		||||
    $sql['PostgreSQL'] = 'ALTER TABLE '.$table.' ADD COLUMN "md5" VARCHAR(32) DEFAULT \'\'';
 | 
			
		||||
    $sql['MySQL'] = 'ALTER TABLE '.$table.' ADD COLUMN `md5` VARCHAR(32) DEFAULT \'\'';
 | 
			
		||||
    $db = Pluf::db();
 | 
			
		||||
    $engine = Pluf::f('db_engine');
 | 
			
		||||
    if (!isset($sql[$engine])) {
 | 
			
		||||
        throw new Exception('SQLite complex migration not supported.');
 | 
			
		||||
    }
 | 
			
		||||
    $db->execute($sql[$engine]);
 | 
			
		||||
    
 | 
			
		||||
    // Process md5 of already uploaded file
 | 
			
		||||
    $files = Pluf::factory('IDF_Upload')->getList();
 | 
			
		||||
    foreach ($files as $f) {
 | 
			
		||||
        $f->md5 = md5_file (Pluf::f('upload_path') . '/' . $f->get_project()->shortname . '/files/' . $f->file);
 | 
			
		||||
        $f->update();
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function IDF_Migrations_18DownloadMD5_down($params=null)
 | 
			
		||||
{
 | 
			
		||||
    // Remove the row
 | 
			
		||||
    $table = Pluf::factory('IDF_Upload')->getSqlTable();
 | 
			
		||||
    $sql = array();
 | 
			
		||||
    $sql['PostgreSQL'] = 'ALTER TABLE '.$table.' DROP COLUMN "md5"';
 | 
			
		||||
    $sql['MySQL'] = 'ALTER TABLE '.$table.' DROP COLUMN `md5`';
 | 
			
		||||
    $db = Pluf::db();
 | 
			
		||||
    $engine = Pluf::f('db_engine');
 | 
			
		||||
    if (!isset($sql[$engine])) {
 | 
			
		||||
        throw new Exception('SQLite complex migration not supported.');
 | 
			
		||||
    }
 | 
			
		||||
    $db->execute($sql[$engine]);
 | 
			
		||||
}
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -53,6 +53,8 @@ function IDF_Migrations_Backup_run($folder, $name=null)
 | 
			
		||||
                    'IDF_Scm_Cache_Git',
 | 
			
		||||
                    'IDF_Queue',
 | 
			
		||||
                    'IDF_Gconf',
 | 
			
		||||
                    'IDF_EmailAddress',
 | 
			
		||||
                    'IDF_IssueRelation',
 | 
			
		||||
                    );
 | 
			
		||||
    $db = Pluf::db();
 | 
			
		||||
    // Now, for each table, we dump the content in json, this is a
 | 
			
		||||
@@ -98,6 +100,8 @@ function IDF_Migrations_Backup_restore($folder, $name)
 | 
			
		||||
                    'IDF_Scm_Cache_Git',
 | 
			
		||||
                    'IDF_Queue',
 | 
			
		||||
                    'IDF_Gconf',
 | 
			
		||||
                    'IDF_EmailAddress',
 | 
			
		||||
                    'IDF_IssueRelation',
 | 
			
		||||
                    );
 | 
			
		||||
    $db = Pluf::db();
 | 
			
		||||
    $schema = new Pluf_DB_Schema($db);
 | 
			
		||||
@@ -110,4 +114,4 @@ function IDF_Migrations_Backup_restore($folder, $name)
 | 
			
		||||
        Pluf_Test_Fixture::load($data, false);
 | 
			
		||||
    }
 | 
			
		||||
    return true;
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -51,6 +51,7 @@ function IDF_Migrations_Install_setup($params=null)
 | 
			
		||||
                    'IDF_Queue',
 | 
			
		||||
                    'IDF_Gconf',
 | 
			
		||||
                    'IDF_EmailAddress',
 | 
			
		||||
                    'IDF_IssueRelation',
 | 
			
		||||
                    );
 | 
			
		||||
    $db = Pluf::db();
 | 
			
		||||
    $schema = new Pluf_DB_Schema($db);
 | 
			
		||||
@@ -109,6 +110,7 @@ function IDF_Migrations_Install_teardown($params=null)
 | 
			
		||||
                    'IDF_Commit',
 | 
			
		||||
                    'IDF_Project',
 | 
			
		||||
                    'IDF_EmailAddress',
 | 
			
		||||
                    'IDF_IssueRelation',
 | 
			
		||||
                    );
 | 
			
		||||
    $db = Pluf::db();
 | 
			
		||||
    $schema = new Pluf_DB_Schema($db);
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -59,6 +59,16 @@ class IDF_Plugin_SyncGit_Cron
 | 
			
		||||
                $out .= sprintf($template, $cmd, $key->login, $content)."\n";
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $out = "# indefero start" . PHP_EOL . $out . "# indefero end" . PHP_EOL;
 | 
			
		||||
        
 | 
			
		||||
        // We update only the part of the file between IDF_START / IDF_END comment
 | 
			
		||||
        $original_keys = file_get_contents($authorized_keys);
 | 
			
		||||
        if (strstr($original_keys, "# indefero start") && strstr($original_keys, "# indefero end")) {
 | 
			
		||||
            $out = preg_replace('/(#\sindefero\sstart).+(#\sindefero\send\s\s?)/isU', 
 | 
			
		||||
                                $out, $original_keys);
 | 
			
		||||
        } else {
 | 
			
		||||
             $out .= $original_keys;   
 | 
			
		||||
        }
 | 
			
		||||
        file_put_contents($authorized_keys, $out, LOCK_EX);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -210,22 +210,23 @@ class IDF_Plugin_SyncGit_Serve
 | 
			
		||||
        // Indefero's one.
 | 
			
		||||
        $p = realpath(dirname(__FILE__).'/../../../../scripts/git-post-update');
 | 
			
		||||
        $p = Pluf::f('idf_plugin_syncgit_post_update', $p);
 | 
			
		||||
        if (!@unlink($fullpath.'/hooks/post-update')) {
 | 
			
		||||
        $post_update_hook = $fullpath.'/hooks/post-update';
 | 
			
		||||
        if (file_exists($post_update_hook) && !@unlink($post_update_hook)) {
 | 
			
		||||
            Pluf_Log::warn(array('IDF_Plugin_Git_Serve::initRepository', 
 | 
			
		||||
                                 'post-update hook removal error.', 
 | 
			
		||||
                                 $fullpath.'/hooks/post-update'));
 | 
			
		||||
                                 $post_update_hook));
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        $out = array();
 | 
			
		||||
        $res = 0;
 | 
			
		||||
        exec(sprintf(Pluf::f('idf_exec_cmd_prefix', '').'ln -s %s %s', 
 | 
			
		||||
                     escapeshellarg($p), 
 | 
			
		||||
                     escapeshellarg($fullpath.'/hooks/post-update')),
 | 
			
		||||
                     escapeshellarg($post_update_hook)),
 | 
			
		||||
             $out, $res);
 | 
			
		||||
        if ($res != 0) {
 | 
			
		||||
            Pluf_Log::warn(array('IDF_Plugin_Git_Serve::initRepository', 
 | 
			
		||||
                                 'post-update hook creation error.', 
 | 
			
		||||
                                 $fullpath.'/hooks/post-update'));
 | 
			
		||||
                                 $post_update_hook));
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        Pluf_Log::debug(array('IDF_Plugin_Git_Serve::initRepository', 
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2010 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -27,6 +27,18 @@
 | 
			
		||||
 */
 | 
			
		||||
class IDF_Plugin_SyncMonotone
 | 
			
		||||
{
 | 
			
		||||
    private $old_err_rep = 0;
 | 
			
		||||
 | 
			
		||||
    public function __construct()
 | 
			
		||||
    {
 | 
			
		||||
        $this->old_err_rep = error_reporting(0);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function __destruct()
 | 
			
		||||
    {
 | 
			
		||||
        error_reporting($this->old_err_rep);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Entry point of the plugin.
 | 
			
		||||
     */
 | 
			
		||||
@@ -80,24 +92,33 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // This guard cleans up on any kind of error, and here is how it works:
 | 
			
		||||
        // As long as the guard is not committed, it keeps a reference to
 | 
			
		||||
        // the given project. When the guard is destroyed and the reference
 | 
			
		||||
        // is still present, it deletes the object. The deletion indirectly
 | 
			
		||||
        // also calls into this plugin again, as the project delete hook
 | 
			
		||||
        // will be called, that removes any changes we've made during the
 | 
			
		||||
        // process.
 | 
			
		||||
        $projectGuard = new IDF_Plugin_SyncMonotone_ModelGuard($project);
 | 
			
		||||
 | 
			
		||||
        $projecttempl = Pluf::f('mtn_repositories', false);
 | 
			
		||||
        if ($projecttempl === false) {
 | 
			
		||||
            throw new IDF_Scm_Exception(
 | 
			
		||||
                 __('"mtn_repositories" must be defined in your configuration file.')
 | 
			
		||||
            $this->_diagnoseProblem(
 | 
			
		||||
                 __('"mtn_repositories" must be defined in your configuration file')
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $usher_config = Pluf::f('mtn_usher_conf', false);
 | 
			
		||||
        if (!$usher_config || !is_writable($usher_config)) {
 | 
			
		||||
            throw new IDF_Scm_Exception(
 | 
			
		||||
                 __('"mtn_usher_conf" does not exist or is not writable.')
 | 
			
		||||
            $this->_diagnoseProblem(
 | 
			
		||||
                 __('"mtn_usher_conf" does not exist or is not writable')
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $mtnpostpush = realpath(dirname(__FILE__) . '/../../../scripts/mtn-post-push');
 | 
			
		||||
        if (!file_exists($mtnpostpush)) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('Could not find mtn-post-push script "%s".'), $mtnpostpush
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('Could not find mtn-post-push script "%s"'), $mtnpostpush
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
@@ -110,13 +131,12 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
             'monotonerc.in',
 | 
			
		||||
             'remote-automate-permissions.in',
 | 
			
		||||
             'hooks.d/',
 | 
			
		||||
             // this is linked and not copied to be able to update
 | 
			
		||||
             // the list of read-only commands on upgrades
 | 
			
		||||
             'hooks.d/indefero_authorize_remote_automate.conf',
 | 
			
		||||
             'hooks.d/indefero_authorize_remote_automate.lua',
 | 
			
		||||
             'hooks.d/indefero_post_push.conf.in',
 | 
			
		||||
             'hooks.d/indefero_post_push.lua',
 | 
			
		||||
        );
 | 
			
		||||
        // enable remote command execution of read-only commands
 | 
			
		||||
        // only for public projects
 | 
			
		||||
        if (!$project->private) {
 | 
			
		||||
            // this is linked and not copied to be able to update
 | 
			
		||||
            // the list of read-only commands on upgrades
 | 
			
		||||
@@ -131,8 +151,8 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        }
 | 
			
		||||
        foreach ($confdir_contents as $content) {
 | 
			
		||||
            if (!file_exists($confdir.$content)) {
 | 
			
		||||
                throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                    __('The configuration file %s is missing.'), $content
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('The configuration file "%s" is missing'), $content
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
@@ -140,14 +160,15 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        $shortname = $project->shortname;
 | 
			
		||||
        $projectpath = sprintf($projecttempl, $shortname);
 | 
			
		||||
        if (file_exists($projectpath)) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('The project path %s already exists.'), $projectpath
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('The project path "%s" already exists'), $projectpath
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (!mkdir($projectpath)) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('The project path %s could not be created.'), $projectpath
 | 
			
		||||
        if (!@mkdir($projectpath)) {
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('The project path "%s" could not be created'),
 | 
			
		||||
                $projectpath
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
@@ -156,7 +177,7 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        //
 | 
			
		||||
        $dbfile = $projectpath.'/database.mtn';
 | 
			
		||||
        $cmd = sprintf('db init -d %s', escapeshellarg($dbfile));
 | 
			
		||||
        self::_mtn_exec($cmd);
 | 
			
		||||
        $this->_mtn_exec($cmd);
 | 
			
		||||
 | 
			
		||||
        //
 | 
			
		||||
        // step 2) create a server key
 | 
			
		||||
@@ -175,16 +196,17 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            escapeshellarg($projectpath),
 | 
			
		||||
            escapeshellarg($serverkey)
 | 
			
		||||
        );
 | 
			
		||||
        self::_mtn_exec($cmd);
 | 
			
		||||
        $this->_mtn_exec($cmd);
 | 
			
		||||
 | 
			
		||||
        //
 | 
			
		||||
        // step 3) create a client key, and save it in IDF
 | 
			
		||||
        //
 | 
			
		||||
        $keydir = Pluf::f('tmp_folder').'/mtn-client-keys';
 | 
			
		||||
        if (!file_exists($keydir)) {
 | 
			
		||||
            if (!mkdir($keydir)) {
 | 
			
		||||
                throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                    __('The key directory %s could not be created.'), $keydir
 | 
			
		||||
            if (!@mkdir($keydir)) {
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('The key directory "%s" could not be created'),
 | 
			
		||||
                    $keydir
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
@@ -194,14 +216,14 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            escapeshellarg($keydir),
 | 
			
		||||
            escapeshellarg($clientkey_name)
 | 
			
		||||
        );
 | 
			
		||||
        $keyinfo = self::_mtn_exec($cmd);
 | 
			
		||||
        $keyinfo = $this->_mtn_exec($cmd);
 | 
			
		||||
 | 
			
		||||
        $parsed_keyinfo = array();
 | 
			
		||||
        try {
 | 
			
		||||
            $parsed_keyinfo = IDF_Scm_Monotone_BasicIO::parse($keyinfo);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception $e) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('Could not parse key information: %s'), $e->getMessage()
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
@@ -219,13 +241,13 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            escapeshellarg($keydir),
 | 
			
		||||
            escapeshellarg($clientkey_hash)
 | 
			
		||||
        );
 | 
			
		||||
        $clientkey_pubdata = self::_mtn_exec($cmd);
 | 
			
		||||
        $clientkey_pubdata = $this->_mtn_exec($cmd);
 | 
			
		||||
 | 
			
		||||
        $cmd = sprintf('au put_public_key --db=%s %s',
 | 
			
		||||
            escapeshellarg($dbfile),
 | 
			
		||||
            escapeshellarg($clientkey_pubdata)
 | 
			
		||||
        );
 | 
			
		||||
        self::_mtn_exec($cmd);
 | 
			
		||||
        $this->_mtn_exec($cmd);
 | 
			
		||||
 | 
			
		||||
        //
 | 
			
		||||
        // step 4) setup the configuration
 | 
			
		||||
@@ -238,18 +260,20 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        foreach ($confdir_contents as $content) {
 | 
			
		||||
            $filepath = $projectpath.'/'.$content;
 | 
			
		||||
            if (substr($content, -1) == '/') {
 | 
			
		||||
                if (!mkdir($filepath)) {
 | 
			
		||||
                    throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                        __('Could not create configuration directory "%s"'), $filepath
 | 
			
		||||
                if (!@mkdir($filepath)) {
 | 
			
		||||
                    $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                        __('Could not create configuration directory "%s"'),
 | 
			
		||||
                        $filepath
 | 
			
		||||
                    ));
 | 
			
		||||
                }
 | 
			
		||||
                continue;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (substr($content, -3) != '.in') {
 | 
			
		||||
                if (!symlink($confdir.$content, $filepath)) {
 | 
			
		||||
                    IDF_Scm_Exception(sprintf(
 | 
			
		||||
                        __('Could not create symlink "%s"'), $filepath
 | 
			
		||||
                if (!@symlink($confdir.$content, $filepath)) {
 | 
			
		||||
                    $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                        __('Could not create symlink for configuration file "%s"'),
 | 
			
		||||
                        $filepath
 | 
			
		||||
                    ));
 | 
			
		||||
                }
 | 
			
		||||
                continue;
 | 
			
		||||
@@ -264,9 +288,10 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
            // remove the .in
 | 
			
		||||
            $filepath = substr($filepath, 0, -3);
 | 
			
		||||
            if (file_put_contents($filepath, $filecontents, LOCK_EX) === false) {
 | 
			
		||||
                throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                    __('Could not write configuration file "%s"'), $filepath
 | 
			
		||||
            if (@file_put_contents($filepath, $filecontents, LOCK_EX) === false) {
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('Could not write configuration file "%s"'),
 | 
			
		||||
                    $filepath
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
@@ -280,7 +305,7 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            $parsed_config = IDF_Scm_Monotone_BasicIO::parse($usher_rc);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception $e) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('Could not parse usher configuration in "%s": %s'),
 | 
			
		||||
                $usher_config, $e->getMessage()
 | 
			
		||||
            ));
 | 
			
		||||
@@ -291,7 +316,7 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            foreach ($stanzas as $stanza_line) {
 | 
			
		||||
                if ($stanza_line['key'] == 'server' &&
 | 
			
		||||
                    $stanza_line['values'][0] == $shortname) {
 | 
			
		||||
                    throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                    $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                        __('usher configuration already contains a server '.
 | 
			
		||||
                           'entry named "%s"'),
 | 
			
		||||
                        $shortname
 | 
			
		||||
@@ -315,9 +340,10 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
        // FIXME: more sanity - what happens on failing writes? we do not
 | 
			
		||||
        // have a backup copy of usher.conf around...
 | 
			
		||||
        if (file_put_contents($usher_config, $usher_rc, LOCK_EX) === false) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('Could not write usher configuration file "%s"'), $usher_config
 | 
			
		||||
        if (@file_put_contents($usher_config, $usher_rc, LOCK_EX) === false) {
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('Could not write usher configuration file "%s"'),
 | 
			
		||||
                $usher_config
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
@@ -325,6 +351,9 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        // step 6) reload usher to pick up the new configuration
 | 
			
		||||
        //
 | 
			
		||||
        IDF_Scm_Monotone_Usher::reload();
 | 
			
		||||
 | 
			
		||||
        // commit the guard, so the newly created project is not deleted
 | 
			
		||||
        $projectGuard->commit();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
@@ -345,8 +374,8 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        $mtn = IDF_Scm_Monotone::factory($project);
 | 
			
		||||
        $stdio = $mtn->getStdio();
 | 
			
		||||
 | 
			
		||||
        $projectpath = self::_get_project_path($project);
 | 
			
		||||
        $auth_ids    = self::_get_authorized_user_ids($project);
 | 
			
		||||
        $projectpath = $this->_get_project_path($project);
 | 
			
		||||
        $auth_ids    = $this->_get_authorized_user_ids($project);
 | 
			
		||||
        $key_ids     = array();
 | 
			
		||||
        foreach ($auth_ids as $auth_id) {
 | 
			
		||||
            $sql = new Pluf_SQL('user=%s', array($auth_id));
 | 
			
		||||
@@ -361,9 +390,10 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
        $write_permissions = implode("\n", $key_ids);
 | 
			
		||||
        $rcfile = $projectpath.'/write-permissions';
 | 
			
		||||
        if (file_put_contents($rcfile, $write_permissions, LOCK_EX) === false) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('Could not write write-permissions file "%s"'), $rcfile
 | 
			
		||||
        if (@file_put_contents($rcfile, $write_permissions, LOCK_EX) === false) {
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('Could not write write-permissions file "%s"'),
 | 
			
		||||
                $rcfile
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
@@ -382,11 +412,13 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
                array('key' => 'allow', 'values' => array('*')),
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $read_permissions = IDF_Scm_Monotone_BasicIO::compile(array($stanza));
 | 
			
		||||
        $rcfile = $projectpath.'/read-permissions';
 | 
			
		||||
        if (file_put_contents($rcfile, $read_permissions, LOCK_EX) === false) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('Could not write read-permissions file "%s"'), $rcfile
 | 
			
		||||
        if (@file_put_contents($rcfile, $read_permissions, LOCK_EX) === false) {
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('Could not write read-permissions file "%s"'),
 | 
			
		||||
                $rcfile
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
@@ -401,16 +433,16 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
        $serverRestartRequired = false;
 | 
			
		||||
        if ($project->private && file_exists($projectfile) && is_link($projectfile)) {
 | 
			
		||||
            if (!unlink($projectfile)) {
 | 
			
		||||
                IDF_Scm_Exception(sprintf(
 | 
			
		||||
            if (!@unlink($projectfile)) {
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('Could not remove symlink "%s"'), $projectfile
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
            $serverRestartRequired = true;
 | 
			
		||||
        } else
 | 
			
		||||
        if (!$project->private && !file_exists($projectfile)) {
 | 
			
		||||
            if (!symlink($templatefile, $projectfile)) {
 | 
			
		||||
                throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
            if (!@symlink($templatefile, $projectfile)) {
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('Could not create symlink "%s"'), $projectfile
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
@@ -422,6 +454,9 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            // seems to be ignored when the server should be started
 | 
			
		||||
            // again immediately afterwards
 | 
			
		||||
            IDF_Scm_Monotone_Usher::killServer($project->shortname);
 | 
			
		||||
            // give usher some time to cool down, otherwise it might hang
 | 
			
		||||
            // (see https://code.monotone.ca/p/contrib/issues/175/)
 | 
			
		||||
            sleep(2);
 | 
			
		||||
            IDF_Scm_Monotone_Usher::startServer($project->shortname);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
@@ -443,8 +478,8 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
        $usher_config = Pluf::f('mtn_usher_conf', false);
 | 
			
		||||
        if (!$usher_config || !is_writable($usher_config)) {
 | 
			
		||||
            throw new IDF_Scm_Exception(
 | 
			
		||||
                 __('"mtn_usher_conf" does not exist or is not writable.')
 | 
			
		||||
            $this->_diagnoseProblem(
 | 
			
		||||
                 __('"mtn_usher_conf" does not exist or is not writable')
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
@@ -453,16 +488,16 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
        $projecttempl = Pluf::f('mtn_repositories', false);
 | 
			
		||||
        if ($projecttempl === false) {
 | 
			
		||||
            throw new IDF_Scm_Exception(
 | 
			
		||||
                 __('"mtn_repositories" must be defined in your configuration file.')
 | 
			
		||||
            $this->_diagnoseProblem(
 | 
			
		||||
                 __('"mtn_repositories" must be defined in your configuration file')
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $projectpath = sprintf($projecttempl, $shortname);
 | 
			
		||||
        if (file_exists($projectpath)) {
 | 
			
		||||
            if (!self::_delete_recursive($projectpath)) {
 | 
			
		||||
                throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                    __('One or more paths underknees %s could not be deleted.'), $projectpath
 | 
			
		||||
            if (!$this->_delete_recursive($projectpath)) {
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('One or more paths underneath %s could not be deleted'), $projectpath
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
@@ -473,8 +508,9 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        if ($keyname && $keyhash &&
 | 
			
		||||
            file_exists($keydir .'/'. $keyname . '.' . $keyhash)) {
 | 
			
		||||
            if (!@unlink($keydir .'/'. $keyname . '.' . $keyhash)) {
 | 
			
		||||
                throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                    __('Could not delete client private key %s'), $keyname
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('Could not delete client private key "%s"'),
 | 
			
		||||
                    $keyname
 | 
			
		||||
                ));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
@@ -485,7 +521,7 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            $parsed_config = IDF_Scm_Monotone_BasicIO::parse($usher_rc);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception $e) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('Could not parse usher configuration in "%s": %s'),
 | 
			
		||||
                $usher_config, $e->getMessage()
 | 
			
		||||
            ));
 | 
			
		||||
@@ -505,9 +541,10 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
        // FIXME: more sanity - what happens on failing writes? we do not
 | 
			
		||||
        // have a backup copy of usher.conf around...
 | 
			
		||||
        if (file_put_contents($usher_config, $usher_rc, LOCK_EX) === false) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('Could not write usher configuration file "%s"'), $usher_config
 | 
			
		||||
        if (@file_put_contents($usher_config, $usher_rc, LOCK_EX) === false) {
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('Could not write usher configuration file "%s"'),
 | 
			
		||||
                $usher_config
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
@@ -528,6 +565,8 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $keyGuard = new IDF_Plugin_SyncMonotone_ModelGuard($key);
 | 
			
		||||
 | 
			
		||||
        foreach (Pluf::factory('IDF_Project')->getList() as $project) {
 | 
			
		||||
            $conf = new IDF_Conf();
 | 
			
		||||
            $conf->setProject($project);
 | 
			
		||||
@@ -535,8 +574,8 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            if ($scm != 'mtn')
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
            $projectpath = self::_get_project_path($project);
 | 
			
		||||
            $auth_ids    = self::_get_authorized_user_ids($project);
 | 
			
		||||
            $projectpath = $this->_get_project_path($project);
 | 
			
		||||
            $auth_ids    = $this->_get_authorized_user_ids($project);
 | 
			
		||||
            if (!in_array($key->user, $auth_ids))
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
@@ -556,7 +595,7 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
                    $parsed_read_perms = IDF_Scm_Monotone_BasicIO::parse($read_perms);
 | 
			
		||||
                }
 | 
			
		||||
                catch (Exception $e) {
 | 
			
		||||
                    throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                    $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                        __('Could not parse read-permissions for project "%s": %s'),
 | 
			
		||||
                        $shortname, $e->getMessage()
 | 
			
		||||
                    ));
 | 
			
		||||
@@ -598,10 +637,11 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
                $read_perms = IDF_Scm_Monotone_BasicIO::compile($parsed_read_perms);
 | 
			
		||||
 | 
			
		||||
                if (file_put_contents($projectpath.'/read-permissions',
 | 
			
		||||
                if (@file_put_contents($projectpath.'/read-permissions',
 | 
			
		||||
                                      $read_perms, LOCK_EX) === false) {
 | 
			
		||||
                    throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                        __('Could not write read-permissions for project "%s"'), $shortname
 | 
			
		||||
                    $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                        __('Could not write read-permissions for project "%s"'),
 | 
			
		||||
                        $shortname
 | 
			
		||||
                    ));
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -611,9 +651,9 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            if (!in_array('*', $lines) && !in_array($mtn_key_id, $lines)) {
 | 
			
		||||
                $lines[] = $mtn_key_id;
 | 
			
		||||
            }
 | 
			
		||||
            if (file_put_contents($projectpath.'/write-permissions',
 | 
			
		||||
            if (@file_put_contents($projectpath.'/write-permissions',
 | 
			
		||||
                                  implode("\n", $lines) . "\n", LOCK_EX) === false) {
 | 
			
		||||
                throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('Could not write write-permissions file for project "%s"'),
 | 
			
		||||
                    $shortname
 | 
			
		||||
                ));
 | 
			
		||||
@@ -623,6 +663,8 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            $stdio = $mtn->getStdio();
 | 
			
		||||
            $stdio->exec(array('put_public_key', $key->content));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $keyGuard->commit();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
@@ -651,8 +693,8 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            if ($scm != 'mtn')
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
            $projectpath = self::_get_project_path($project);
 | 
			
		||||
            $auth_ids    = self::_get_authorized_user_ids($project);
 | 
			
		||||
            $projectpath = $this->_get_project_path($project);
 | 
			
		||||
            $auth_ids    = $this->_get_authorized_user_ids($project);
 | 
			
		||||
            if (!in_array($key->user, $auth_ids))
 | 
			
		||||
                continue;
 | 
			
		||||
 | 
			
		||||
@@ -672,7 +714,7 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
                    $parsed_read_perms = IDF_Scm_Monotone_BasicIO::parse($read_perms);
 | 
			
		||||
                }
 | 
			
		||||
                catch (Exception $e) {
 | 
			
		||||
                    throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                    $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                        __('Could not parse read-permissions for project "%s": %s'),
 | 
			
		||||
                        $shortname, $e->getMessage()
 | 
			
		||||
                    ));
 | 
			
		||||
@@ -693,10 +735,11 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
 | 
			
		||||
                $read_perms = IDF_Scm_Monotone_BasicIO::compile($parsed_read_perms);
 | 
			
		||||
 | 
			
		||||
                if (file_put_contents($projectpath.'/read-permissions',
 | 
			
		||||
                if (@file_put_contents($projectpath.'/read-permissions',
 | 
			
		||||
                                      $read_perms, LOCK_EX) === false) {
 | 
			
		||||
                    throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                        __('Could not write read-permissions for project "%s"'), $shortname
 | 
			
		||||
                    $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                        __('Could not write read-permissions for project "%s"'),
 | 
			
		||||
                        $shortname
 | 
			
		||||
                    ));
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -711,9 +754,9 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            if (file_put_contents($projectpath.'/write-permissions',
 | 
			
		||||
                                  implode("\n", $lines) . "\n", LOCK_EX) === false) {
 | 
			
		||||
                throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
            if (@file_put_contents($projectpath.'/write-permissions',
 | 
			
		||||
                                   implode("\n", $lines) . "\n", LOCK_EX) === false) {
 | 
			
		||||
                $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                    __('Could not write write-permissions file for project "%s"'),
 | 
			
		||||
                    $shortname
 | 
			
		||||
                ));
 | 
			
		||||
@@ -762,7 +805,43 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        ));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private static function _get_authorized_user_ids($project)
 | 
			
		||||
    private function _get_project_path($project)
 | 
			
		||||
    {
 | 
			
		||||
        $projecttempl = Pluf::f('mtn_repositories', false);
 | 
			
		||||
        if ($projecttempl === false) {
 | 
			
		||||
            $this->_diagnoseProblem(
 | 
			
		||||
                 __('"mtn_repositories" must be defined in your configuration file.')
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $projectpath = sprintf($projecttempl, $project->shortname);
 | 
			
		||||
        if (!file_exists($projectpath)) {
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('The project path %s does not exists.'), $projectpath
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
        return $projectpath;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private function _mtn_exec($cmd)
 | 
			
		||||
    {
 | 
			
		||||
        $fullcmd = sprintf('%s %s %s',
 | 
			
		||||
            Pluf::f('idf_exec_cmd_prefix', ''),
 | 
			
		||||
            Pluf::f('mtn_path', 'mtn'),
 | 
			
		||||
            $cmd
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        $output = $return = null;
 | 
			
		||||
        exec($fullcmd, $output, $return);
 | 
			
		||||
        if ($return != 0) {
 | 
			
		||||
            $this->_diagnoseProblem(sprintf(
 | 
			
		||||
                __('The command "%s" could not be executed.'), $cmd
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
        return implode("\n", $output);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private function _get_authorized_user_ids($project)
 | 
			
		||||
    {
 | 
			
		||||
        $mem = $project->getMembershipData();
 | 
			
		||||
        $members = array_merge((array)$mem['members'],
 | 
			
		||||
@@ -775,43 +854,7 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
        return $userids;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private static function _get_project_path($project)
 | 
			
		||||
    {
 | 
			
		||||
        $projecttempl = Pluf::f('mtn_repositories', false);
 | 
			
		||||
        if ($projecttempl === false) {
 | 
			
		||||
            throw new IDF_Scm_Exception(
 | 
			
		||||
                 __('"mtn_repositories" must be defined in your configuration file.')
 | 
			
		||||
            );
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $projectpath = sprintf($projecttempl, $project->shortname);
 | 
			
		||||
        if (!file_exists($projectpath)) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('The project path %s does not exists.'), $projectpath
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
        return $projectpath;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private static function _mtn_exec($cmd)
 | 
			
		||||
    {
 | 
			
		||||
        $fullcmd = sprintf('%s %s %s',
 | 
			
		||||
            Pluf::f('idf_exec_cmd_prefix', ''),
 | 
			
		||||
            Pluf::f('mtn_path', 'mtn'),
 | 
			
		||||
            $cmd
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        $output = $return = null;
 | 
			
		||||
        exec($fullcmd, $output, $return);
 | 
			
		||||
        if ($return != 0) {
 | 
			
		||||
            throw new IDF_Scm_Exception(sprintf(
 | 
			
		||||
                __('The command "%s" could not be executed.'), $cmd
 | 
			
		||||
            ));
 | 
			
		||||
        }
 | 
			
		||||
        return implode("\n", $output);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private static function _delete_recursive($path)
 | 
			
		||||
    private function _delete_recursive($path)
 | 
			
		||||
    {
 | 
			
		||||
        if (is_file($path) || is_link($path)) {
 | 
			
		||||
            return @unlink($path);
 | 
			
		||||
@@ -821,10 +864,48 @@ class IDF_Plugin_SyncMonotone
 | 
			
		||||
            $scan = glob(rtrim($path, '/') . '/*');
 | 
			
		||||
            $status = 0;
 | 
			
		||||
            foreach ($scan as $subpath) {
 | 
			
		||||
                $status |= self::_delete_recursive($subpath);
 | 
			
		||||
                $status |= $this->_delete_recursive($subpath);
 | 
			
		||||
            }
 | 
			
		||||
            $status |= rmdir($path);
 | 
			
		||||
            $status |= @rmdir($path);
 | 
			
		||||
            return $status;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private function _diagnoseProblem($msg)
 | 
			
		||||
    {
 | 
			
		||||
        $system_err = error_get_last();
 | 
			
		||||
        if (!empty($system_err)) {
 | 
			
		||||
            $msg .= ': '.$system_err['message'];
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        error_reporting($this->old_err_rep);
 | 
			
		||||
        throw new IDF_Scm_Exception($msg);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A simple helper class that deletes the model instance if
 | 
			
		||||
 * it is not committed
 | 
			
		||||
 */
 | 
			
		||||
class IDF_Plugin_SyncMonotone_ModelGuard
 | 
			
		||||
{
 | 
			
		||||
    private $model;
 | 
			
		||||
 | 
			
		||||
    public function __construct(Pluf_Model $m)
 | 
			
		||||
    {
 | 
			
		||||
        $this->model = $m;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function __destruct()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->model == null)
 | 
			
		||||
            return;
 | 
			
		||||
        $this->model->delete();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function commit()
 | 
			
		||||
    {
 | 
			
		||||
        $this->model = null;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
-- ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
-- This file is part of InDefero, an open source project management application.
 | 
			
		||||
-- Copyright (C) 2011 Céondo Ltd and contributors.
 | 
			
		||||
-- Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
-- Copyright (C) 2010 Thomas Keller <me@thomaskeller.biz>
 | 
			
		||||
--                    Richard Levitte <richard@levitte.org>
 | 
			
		||||
--
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
-- ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
-- This file is part of InDefero, an open source project management application.
 | 
			
		||||
-- Copyright (C) 2011 Céondo Ltd and contributors.
 | 
			
		||||
-- Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
--
 | 
			
		||||
-- InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
-- it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
-- ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
-- This file is part of InDefero, an open source project management application.
 | 
			
		||||
-- Copyright (C) 2011 Céondo Ltd and contributors.
 | 
			
		||||
-- Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
--
 | 
			
		||||
-- InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
-- it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -140,7 +140,48 @@ class IDF_Project extends Pluf_Model
 | 
			
		||||
     * @param IDF_Tag Subfilter with a label (null)
 | 
			
		||||
     * @return int Count
 | 
			
		||||
     */
 | 
			
		||||
    public function getIssueCountByStatus($status='open', $label=null)
 | 
			
		||||
    public function getIssueCountByOwner($status='open')
 | 
			
		||||
    {
 | 
			
		||||
        switch ($status) {
 | 
			
		||||
        case 'open':
 | 
			
		||||
            $tags = implode(',', $this->getTagIdsByStatus('open'));
 | 
			
		||||
            break;
 | 
			
		||||
        case 'closed':
 | 
			
		||||
        default:
 | 
			
		||||
            $tags = implode(',', $this->getTagIdsByStatus('closed'));
 | 
			
		||||
            break;
 | 
			
		||||
        }
 | 
			
		||||
        $sqlIssueTable = Pluf::factory('IDF_Issue')->getSqlTable();
 | 
			
		||||
        $query = "SELECT uid AS id,COUNT(uid) AS nb
 | 
			
		||||
FROM (
 | 
			
		||||
    SELECT COALESCE(owner, -1) AS uid
 | 
			
		||||
    FROM $sqlIssueTable
 | 
			
		||||
    WHERE status IN ($tags)
 | 
			
		||||
    ) AS ff
 | 
			
		||||
GROUP BY uid";
 | 
			
		||||
 | 
			
		||||
        $db = Pluf::db();
 | 
			
		||||
        $dbData = $db->select($query);
 | 
			
		||||
        $ownerStatistics = array();
 | 
			
		||||
        foreach ($dbData as $k => $v) {
 | 
			
		||||
            $key = ($v['id'] === '-1') ? null : $v['id'];
 | 
			
		||||
            $ownerStatistics[$key] = (int)$v['nb'];
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        arsort($ownerStatistics);
 | 
			
		||||
 | 
			
		||||
        return $ownerStatistics;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Returns the number of open/closed issues.
 | 
			
		||||
     *
 | 
			
		||||
     * @param string Status ('open'), 'closed'
 | 
			
		||||
     * @param IDF_Tag Subfilter with a label (null)
 | 
			
		||||
     * @param array Restrict further to a list of ids
 | 
			
		||||
     * @return int Count
 | 
			
		||||
     */
 | 
			
		||||
    public function getIssueCountByStatus($status='open', $label=null, $ids=array())
 | 
			
		||||
    {
 | 
			
		||||
        switch ($status) {
 | 
			
		||||
        case 'open':
 | 
			
		||||
@@ -163,12 +204,48 @@ class IDF_Project extends Pluf_Model
 | 
			
		||||
            $sql2 = new Pluf_SQL('idf_tag_id=%s', array($label->id));
 | 
			
		||||
            $sql->SAnd($sql2);
 | 
			
		||||
        }
 | 
			
		||||
        if (count($ids) > 0) {
 | 
			
		||||
            $sql2 = new Pluf_SQL(sprintf('id IN (%s)', implode(', ', $ids)));
 | 
			
		||||
            $sql->SAnd($sql2);
 | 
			
		||||
        }
 | 
			
		||||
        $params = array('filter' => $sql->gen());
 | 
			
		||||
        if (!is_null($label)) { $params['view'] = 'join_tags'; }
 | 
			
		||||
        $gissue = new IDF_Issue();
 | 
			
		||||
        return $gissue->getCount($params);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Get the tags for a specific list of issues.
 | 
			
		||||
     *
 | 
			
		||||
     * @param string Status ('open') or 'closed'
 | 
			
		||||
     * @param array A list of issue ids
 | 
			
		||||
     * @return array An array of tag objects
 | 
			
		||||
     */
 | 
			
		||||
    public function getTagsByIssues($issue_ids=array())
 | 
			
		||||
    {
 | 
			
		||||
        // make the below query always a valid one
 | 
			
		||||
        if (count($issue_ids) == 0) $issue_ids[] = 0;
 | 
			
		||||
 | 
			
		||||
        $assocTable = $this->_con->pfx.'idf_issue_idf_tag_assoc';
 | 
			
		||||
        $query = sprintf(
 | 
			
		||||
            'SELECT DISTINCT idf_tag_id FROM %s '.
 | 
			
		||||
            'WHERE idf_issue_id IN (%s) '.
 | 
			
		||||
            'GROUP BY idf_tag_id',
 | 
			
		||||
            $assocTable, implode(',', $issue_ids)
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        $db = Pluf::db();
 | 
			
		||||
        $dbData = $db->select($query);
 | 
			
		||||
        $ids = array(0);
 | 
			
		||||
        foreach ($dbData as $data) {
 | 
			
		||||
            $ids[] = $data['idf_tag_id'];
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $sql = new Pluf_SQL(sprintf('id IN (%s)', implode(', ', $ids)));
 | 
			
		||||
        $model = new IDF_Tag();
 | 
			
		||||
        return $model->getList(array('filter' => $sql->gen()));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Get the open/closed tag ids as they are often used when doing
 | 
			
		||||
     * listings.
 | 
			
		||||
@@ -233,6 +310,29 @@ class IDF_Project extends Pluf_Model
 | 
			
		||||
        return $tags;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Returns a list of relations which are available in this project as
 | 
			
		||||
     * associative array. Each key-value pair marks a set of orthogonal
 | 
			
		||||
     * relations. To ease processing, each of these pairs is included twice
 | 
			
		||||
     * in the array, once as key1 => key2 and once as key2 => key1.
 | 
			
		||||
     *
 | 
			
		||||
     * @return array List of relation names
 | 
			
		||||
     */
 | 
			
		||||
    public function getRelationsFromConfig()
 | 
			
		||||
    {
 | 
			
		||||
        $conf = $this->getConf();
 | 
			
		||||
        $rel = $conf->getVal('issue_relations', IDF_Form_IssueTrackingConf::init_relations);
 | 
			
		||||
        $relations = array();
 | 
			
		||||
        foreach (preg_split("/\015\012|\015|\012/", $rel, -1, PREG_SPLIT_NO_EMPTY) as $s) {
 | 
			
		||||
            $verbs = preg_split("/\s*,\s*/", $s, 2);
 | 
			
		||||
            if (count($verbs) == 1)
 | 
			
		||||
                $relations += array($verbs[0] => $verbs[0]);
 | 
			
		||||
            else
 | 
			
		||||
                $relations += array($verbs[0] => $verbs[1], $verbs[1] => $verbs[0]);
 | 
			
		||||
        }
 | 
			
		||||
        return $relations;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Return membership data.
 | 
			
		||||
     *
 | 
			
		||||
@@ -352,7 +452,11 @@ class IDF_Project extends Pluf_Model
 | 
			
		||||
        foreach ($this->_con->select($sql) as $idc) {
 | 
			
		||||
            $tag = new IDF_Tag($idc['id']);
 | 
			
		||||
            $tag->nb_use = $idc['nb_use'];
 | 
			
		||||
            $tags[] = $tag;
 | 
			
		||||
            // group by class
 | 
			
		||||
            if (!array_key_exists($tag->class, $tags)) {
 | 
			
		||||
                $tags[$tag->class] = array();
 | 
			
		||||
            }
 | 
			
		||||
            $tags[$tag->class][] = $tag;
 | 
			
		||||
        }
 | 
			
		||||
        return new Pluf_Template_ContextVars($tags);
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
@@ -88,22 +88,36 @@ class IDF_Scm
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Run exec and log some information.
 | 
			
		||||
     * Runs the given command and log some information.
 | 
			
		||||
     *
 | 
			
		||||
     * A previous version used plain exec(), but this should not be used
 | 
			
		||||
     * for various reasons, one being that this command does not preserve
 | 
			
		||||
     * trailing whitespace, which is essential for proper diff parsing.
 | 
			
		||||
     *
 | 
			
		||||
     * @param $caller Calling method
 | 
			
		||||
     * @param $cmd Command to run
 | 
			
		||||
     * @param &$out Array of output
 | 
			
		||||
     * @param &$return Return value
 | 
			
		||||
     * @return string Last line of the command
 | 
			
		||||
     */
 | 
			
		||||
    public static function exec($caller, $cmd, &$out=null, &$return=null)
 | 
			
		||||
    {
 | 
			
		||||
        $return = -1;
 | 
			
		||||
        Pluf_Log::stime('timer');
 | 
			
		||||
        $ret = exec($cmd, $out, $return);
 | 
			
		||||
        $fp = popen($cmd, 'r');
 | 
			
		||||
        $buf = '';
 | 
			
		||||
        if ($fp !== false) {
 | 
			
		||||
            while (!feof($fp)) {
 | 
			
		||||
                $buf .= fread($fp, 1024);
 | 
			
		||||
            }
 | 
			
		||||
            $return = pclose($fp);
 | 
			
		||||
        }
 | 
			
		||||
        $out = preg_split('/\r\n|\r|\n/', $buf);
 | 
			
		||||
        $elem = count($out);
 | 
			
		||||
        if ($elem > 0 && $out[$elem-1] === '')
 | 
			
		||||
            unset($out[$elem-1]);
 | 
			
		||||
        Pluf_Log::perf(array($caller, $cmd, Pluf_Log::etime('timer', 'total_exec')));
 | 
			
		||||
        Pluf_Log::debug(array($caller, $cmd, $out));
 | 
			
		||||
        Pluf_Log::inc('exec_calls');
 | 
			
		||||
        return $ret;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
@@ -325,7 +339,8 @@ class IDF_Scm
 | 
			
		||||
     * stdClass object {
 | 
			
		||||
     *  'additions' => array('path/to/file', 'path/to/directory', ...),
 | 
			
		||||
     *  'deletions' => array('path/to/file', 'path/to/directory', ...),
 | 
			
		||||
     *  'renames' => array('old/path/to/file' => 'new/path/to/file', ...)
 | 
			
		||||
     *  'renames' => array('old/path/to/file' => 'new/path/to/file', ...),
 | 
			
		||||
     *  'copies' => array('path/to/source' => 'path/to/target', ...),
 | 
			
		||||
     *  'patches' => array('path/to/file', ...),
 | 
			
		||||
     *  'properties' => array('path/to/file' => array(
 | 
			
		||||
     *              'propname' => 'propvalue', 'deletedprop' => null, ...)
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
/*
 | 
			
		||||
# ***** BEGIN LICENSE BLOCK *****
 | 
			
		||||
# This file is part of InDefero, an open source project management application.
 | 
			
		||||
# Copyright (C) 2008 Céondo Ltd and contributors.
 | 
			
		||||
# Copyright (C) 2008-2011 Céondo Ltd and contributors.
 | 
			
		||||
#
 | 
			
		||||
# InDefero is free software; you can redistribute it and/or modify
 | 
			
		||||
# it under the terms of the GNU General Public License as published by
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user