diff --git a/catalog/fgaddon-catalog/catalog.config.xml b/catalog/fgaddon-catalog/catalog.config.xml
new file mode 100644
index 0000000..e7c15c1
--- /dev/null
+++ b/catalog/fgaddon-catalog/catalog.config.xml
@@ -0,0 +1,25 @@
+
+
+
+ /home/curt/Projects/FlightGear/ftp/Aircraft
+ http://mirrors.ibiblio.org/flightgear/ftp/Aircraft/
+ http://mirrors.ibiblio.org/flightgear/ftp/Aircraft/thumbnails
+
+ svn
+ /home/curt/Projects/FlightGear/flightgear-fgaddon/Aircraft
+ NTPS
+ c172
+ tu134
+
+
+
+ true
+ rsync-ssh
+ fgfs:/home/fgfs/fgfs.goneabitbursar.com/official
+
+
diff --git a/catalog/fgaddon-catalog/md5sum.xml b/catalog/fgaddon-catalog/md5sum.xml
new file mode 100644
index 0000000..6d3cbdf
--- /dev/null
+++ b/catalog/fgaddon-catalog/md5sum.xml
@@ -0,0 +1,516 @@
+
+
+ 8a4ea48a3bb83b608386f942a85100c9
+ 057bfdb3c3fcfba3f8d31723a77df277
+ 088f6c3850d9c3f8e94ced2bfc2a65db
+ abc393890f7ab49f32c6a2adc01b7823
+ 8e93ee83c42f77db9881f9ca0f00cf8b
+ 45fc009e9323de75e477b2548c79cb9a
+ 3fb8820c373e78bcd52981293e2d46e8
+ c974d9b2e7c5feba195563433a4c5bc2
+ dd9ab7cb3fb27a4acfc7b634fa9ec396
+ 5779ec5bdab4e9405d0db447c5a73df4
+ e3565e459436a5e4dcdd260a095c28b2
+ ce6c904cb23df6337ed6bdc1dcf6d68e
+ 92161d0ef86048c0c859241abd64ffcd
+ 9f5841317f4ed65fc815517c6770affd
+ 89c1b762b8b58fee4772a94d3c9b81d8
+ 693bf6470439f876edced4bd558b69a1
+ f450b755c5bad640d8c2ef7312b1a6fb
+ 4aaca65794466d8cbf8e9ab29ef1b47b
+ c57f1d877e851f2164f292997508dd36
+ 86b52be36911fd0ee5d0b93d4b70a087
+ 904f23999e33d323d3f7a09c1807b71a
+ 3ae04898384a6531fa2edac2fcd508e9
+ c9ec3df5ce80cbbbec9e39a849877ceb
+ f015a7d2c8bde1224298261ae34e04c2
+ 4951432b9d39c142e891da3209e3ad32
+ f175e90ccf2c3ae5118cf6eb1a43867e
+ dfb5b7f4f7233d854dca45ae4521e080
+ f6757f3a4f78364c0439fd7d488c4330
+ 021be8bd26c7aa03093e5e356d8c7ef2
+ 89c4f1663f6b65ca1f70184dd8886f8b
+ 9049ec76bddd80abff57a837313771d8
+ 4f9c9c24e662036de1cd85ed528c3d4e
+ 766eb03f38f36190de94422b5481ce4b
+ 14219343b589d8610da96f7a8ec16cdc
+ 67689844de14068b7e95e675c0da6c17
+ 44fd70dc27b3d36bbde5766ac528d095
+ dae16af174310b492c4412e0c77593e8
+ 64c7f53c6dac1b4da3813661f7455faf
+ 7ccb293d5808ca0b9c48271a6398de94
+ 61af35090bbc722076353d1496c2d82f
+ edae78dee86c3c0cb741bc4ea8793162
+ 94d5008f7bbfb65a56cff0694983d8ff
+ c267be9f90def2460ff53649c6fee2f0
+ 3fd4701f2363c41de0374c4703b88e83
+ 87a1ad2288b56c71559549d55b93cccd
+ 37ccf863b017f0b773885619a671f8f7
+ 3e26ca3bf5296630eebbcd82859d5ac4
+ add26dd267eca688d4a88b672749a155
+ 75e69854bf9d09d16cbd5147926ba7fd
+ 457f71d5be8ec8c3e42c6919b226b549
+ 4a27ba4bb53f82bf2a4b325cd3405a89
+ b7e3e73c98d17006e984e056442c29ce
+ c0c500c55706fc18d93817ebc3b6e83b
+ c883cde7942cdcfb8cd69c135c4d29f7
+ 70db9ed1cb09e1442e22d020a9dc3069
+ c62f9ee3f78c98a6342a6a337089fd10
+ 264d821437a558cf71c9209f466f6103
+ 27923bdc9f0f3ae3449f5d7a17744da5
+ 023bbe168eb0babb4a68963edffc273b
+ 0926a8a021c4e11fa2610cbb5f6f11b3
+ 846d8156216e434e789f5d2b7432b68f
+ 887c1bc8ab679604031f983ab6c67974
+ 915cd6323c113bb2a8b57e4c08900f3e
+ c0fe4169db9a338f5beec102f1efe301
+ 4b2acc554752df97194e0da1a1c8638c
+ 9d3c4b4b66a7140628b79c4f142cc0f4
+ 611d6172991766b32a43d19bfb06bbb2
+ 7cf94e2ee2c76783bf17c845e12e48cb
+ 6407f6fd1bfe031b280639d20dfe100f
+ 04db71541bc7099383524080a2c64012
+ 2bdd09054c9c92088ff297d6358a05a8
+ 51c6e9397e1e8fff65eb1d1671d18067
+ 3d1883d3213d2b5f7d39ec5221cd4eb2
+ d172dc57ce4e2c15f7f550a5c0c734a8
+ db7301be7b428db2070e27a5fe7ee06e
+ a71912ae798780eba390f9b27638b139
+ 7dd2ae465a7710921d11c6c18aa0c0e7
+ 6aaa660040bd40facf28fc68220b3cfc
+ 2a3387f97647ad875060bac1284cfcbd
+ 46a6f17c638e177421d1afc559b9d66a
+ 50af4a2d301fafb2a3e13fe96e9215be
+ 6395fcaf780d6e43b4dd9f215a24e718
+ 5a133a789694642b9f3e611ddf23d3d6
+ b668b91e67ef10757839da013faee44b
+ 635dc4a39a43814f2885d59b1dda8a9d
+ aba5e103f315014925dff3b39bd2d648
+ ba6e32553464e03fe6392303a2072631
+ 3c10ccdf2775cad485bb394d5b1e56eb
+ 167d03a6eafc1a3780f6fdf733fc7c6e
+ 0579472d3185c8beba310aef9bf65f5d
+ cb732caa641493f07c4cc564018dd481
+ 21b170a0f9dae055d5c1d1af9946515e
+ 4fdcfd3aa209318c07c779e408e81df2
+ aef8c5f5f68349a7ae804df1d6d07e3f
+ e499504fcba1e95e0cca9c4418bfee23
+ a9af2d04a1e7bf279d135a4a06b11181
+ 506a63476d00ef552b8adc3260ba1fa4
+ 9a3fd12ef60228dca05efa4ff694a882
+ 152fcac32e4bb9f7fce5b6894b6da08b
+ 97a640a4f9eddc33ac626f78adfaecb9
+ f00599943616e14682c2baea90e6173a
+ 787d13e37c31fbf894d5a25091e4befc
+ 78d4695d2466f50d3c45e0374809b386
+ 6d0e61e40af698ea72e2978ed50e3a49
+ b3e9444c45cc9c445961b726429c0c7e
+ a230231691ebc4df86e7d645f31a0f07
+ 84362d957e9dae0e5abc70ba4cb9816e
+ 311a00f90c55c01980688c1d5504df64
+ cb75a6990af600c666b02398cf77d6d1
+ a191302bdfb1faff1c6f2dd8a5494381
+ 258b971dd3041a49a816a2150d7e7125
+ 695ce9bf3221055eecd24be01acb57e2
+ 3c341ded96ce7c8832738c34c5c115f8
+ c954a3ab900676c0cbabb1bada93754b
+ 270cabd6289080468d29a807f612cb4c
+ 60bb07c7cefe930145d4b199b63842a7
+ fd61954ffdd247b47fec0fc9074aa40b
+ 171f7d74d45a50804c8111f5fab29543
+ 604cd9887e8f1c9011581348feca1e3c
+ 80756d598e81a6a09217da9025e6284f
+ b465d263b244c5860ce010371032b5d3
+ 60cb1a6effe4e0023edd1bfdc109aed3
+ e2881ea7546775db6847bb3890c04a03
+ 68ec43ed98164bef020452e3758ec22f
+ a865e922bec69cf8a6fe1f8d53dbc71b
+ 4a7cb6a6f5f6d71d5eb2da025b42d71f
+ bd103f5fa8a8ac980b8c4ed3a041ff6b
+ d4eb3933d210d1a8f1deaf8b43a65960
+ 0ad6cdbf91f37b6c71f94268914d2615
+ d94119914089db0f7b6d43ddf20cb302
+ 826b017ad5bc1c77c65959367f41415f
+ 4cc78b3c9c421c382b4e197de6589535
+ 6c8b251c720301a00276f91db736ee85
+ 28ebd3b1236a53a0b15bf9c6f5cc79d6
+ edbaa02c5b837b9522ad8d0fd7da81e8
+ e16b7b135dbe3fa75539931c950116b6
+ bf3b60a808daefd7b7374f934e7b10bf
+ d46d6e08cd90412088abffe2dbd03982
+ 55d564ff1b335a723d144e423b9a3c10
+ a3bbd80a8c11bea16f567eafa898db8f
+ c6f74c5596f959608e1810004abba534
+ a6743e193bb01776107fec98595555e4
+ 8bf718df53a6af7777f087a302641061
+ b66f175629a52396dd1db3f1718d2028
+ 77100ce686eee7ae2b5950df44a38709
+ f132805e11b1c1e60908080b9b6533ad
+ 8b227cd83d5f1f3b3b44bb7b46714349
+ 566f1dc35ae741cdeee125111f1e0c20
+ 633550eae44633a9b23777ffab527f1b
+ d8fa7f026fc404ad20b7c7f6a11619ee
+ bc2267347bc48ba2377926a7a6515274
+ 0d500b113ae2d633e5f0bf36a1d19fd9
+ d935f718bed4a71069122fe5214ecd2e
+ 6062a91295fbf99801e65197a71451c5
+ 8ca56323ab3db01ec034f231490a7702
+ a0d00a78d80e4a621a1cfb55c0658fe3
+ 8b09cc7658164d311f4e70e36f463cde
+ b967ca884dd02b37ad9c247d31bdf1c7
+ f9c33778a3805892fbd0e6de35e54e91
+ 387fd1c59b2e5934ffcd888df39b7174
+ bdc3b2e695db1588827d000b55bbebda
+ 1d37211f993a920b0806b95db0f8ea7c
+ 77f9e6a9e1647dcd012966af7df1d4be
+ dbfd9548db8318098daa75733171a18e
+ 70cff7394ced9ab348653ad82ece3c10
+ 26883edc54c7a02e208ce51e7b964e2e
+ 00fc8120294619b97ccf0e04a6145c1f
+ 9003650773fd4933b40089d146484045
+ 033799e2157ac9ffc6ac04fcdfeafe04
+ 02e81c7a33551c2581f8d5767e464518
+ f7ac35642502a83a265f7b6594650279
+ be3e5fcef3000c72a2d85fb9eb827793
+ c745759b0e259323b20ba2e535284e7d
+ 8b24f65ff28f7c66e65bcf7025582575
+ 14b8d09e2debfdfd9cac81b8b67c9498
+ ff6f3470289772182d931daa00f6057f
+ a0506714e972c2a58f745c6bd7cc4fe2
+ 4a792858aa65aada7cb91a47a9cf8c5d
+ 81fb6e7b8ff50d43180bb758d1a56121
+ cca58285c8c814dfe923aaec656b1d9c
+ 26885d6e6a849ad9dd54829baa26daee
+ e9992fcecbc66640a932c1c8fb7e71ae
+ 8915f9465c563740f9f8f2cf32f1dd74
+ 80757f34ea04417b6b30d65092ce50ab
+ ba8bbc9e04405db298beb9c24c6200c3
+ 41356deabb2cd689393f36b96123a330
+ b57c14a1791a20bccbf23db826a22dbb
+ 578771e25e6d7d81e5574d8cd1d19de0
+ 01c814d7ddeb0adcee0ed58022ebf6a8
+ 259cd05cfbf4b80b2cb4be07cf4bf781
+ 012f8917ef6d7ec50d8a73c0508e5b47
+ 9f77b8fbf981c4d761d8c497b6ef0868
+ 48795402cda50957be7b7504b4d82389
+ d015100dfb5381135d221f41b293584d
+ 894c16ccfa5aed2778c7be4873947b69
+ a99c8114ca48074fea2c675a8e6908e0
+ 684d979f81bf15b2adf47b52d6ca21f7
+ 39631381049db9e54655c499f91bfb31
+ ef5ed515fe8f93876137bb6b27f3b158
+ bdaa474d4be6e51b21266ce0c8782c2c
+ f6ba735798172b43abf417c80c1c4db4
+ 0f915d06528b1cc2dcc16f7be3b9ab37
+ 72a8b154cda42a2ec507b6a4c604afcb
+ 84bfa325c11d56920062e8516bfc146e
+ ba904d06c2ddb28a7fd3f95ac344de7a
+ 3c0963e6203c4a6315155012bbbb8096
+ 43f556905826c6ad5c80af409eb8a76c
+ e1875c6efdc7ffe1160d8e64ac5f5fd7
+ ea67ec6bb6c981dbbd391003f997c6ef
+ 41b4502a287018e27402925094324209
+ 25805162fbd9d4d00e09c814f0beef9b
+ 45264d8293d22533f56f571283940754
+ fb47e4425240141a45818dcdc8060a4b
+ b4fcbc6e73565001d25f3cd64fb85a77
+ 0ea6ad997b8595f97530d9aed5529ec9
+ 9a7c5ac1aabe992abf14fd4b53c1712c
+ 305fc09e91c52fefd2b67d6925baff38
+ 3746de4faf66266399a9273321088ce2
+ ec397cdc0f61d78ecc28d2cc564becd6
+ 3248a3f496cc146fcd95c61ec2538ae7
+ 96e43cd67c0c8b9fd6e158a3ee46fbdc
+ 6d78633d200d8f945be6400027116539
+ 3e0a4af617e2064e78cee25e1d8af653
+ 42f45908ac131a855f66e63c09279562
+ 15f03458e112d4d7d3ff674dc20286c2
+ a3854718af2cf6c4c0512a166e283b4c
+ 24144321fdadaf71e81f8b014ad0e16b
+ 2294512dc1cdb63a8b90a21ba48875c7
+ c56ee7fbdb10b1ba979d019cc26972cf
+ 8bd5f800b24dbfc37f59fe368a748af1
+ f926af126c01eabd3ceb0398b178519a
+ 29cd9e7ea06489595979f3df518813dd
+ ea6025e2b6cdffc86764c5ee5a949c9a
+ c3f68417e01ab653e66131bd446f8940
+ 2fc955ad2ecb4ba0a7899097e7e4b226
+ 83874804ebae4b77f6ac92f55c19ba91
+ da5fc6e9761203e6dd26dc3f01a2fe68
+ 6b17d6d7bc78b5424420d47c0d737fa9
+ 83743f288a09c89716ac2a2e212b11bc
+ 866329f60e90f3404a6bace54e733b73
+ 449953e8d06e61e78e94e520d8fbc3e3
+ 5179bd127ee44112ec06887e322725aa
+ 049759aaa0c615f71b0e1e2d683cb3ba
+ 977f70220ecee03ef7367b940956122a
+ bf993f8a5499538d2b5f5f5ec0c7574d
+ 65223229f5f8adc17c534412fb1b2bc8
+ 9905dc7569bc1e9cf750e47cc138dbeb
+ ef530cb8d0f3b53b83d993f4284bc4c3
+ d18ed659fb34df8e078b36d0156de731
+ 57ada6ce17e8dda569e49ef29a517b90
+ 65c28732f4909889ca9739953c7b2032
+ 82ca3e94ee5a1ee711fbc73e7ab99645
+ b80921cb1ec91813ba6912abf8d3d73c
+ 40f6523b7cc0aff71281820d54bdaa0f
+ 515e08d2a12b2cc1891f23e086aa5933
+ d8250d2d9f842108ffed965ced81ab64
+ 34ea914ec8c9f7c85ba3c243c6d0f981
+ 837d0bc1db372b8d18e251fb63aa13ea
+ 98a417480a3126432d9214fe0efb83c9
+ 568eeccd7ffc3b08937fdf2da81a972a
+ 8f90009d9f8c66ba0dafc1c05ff2c685
+ b3d6038ff416a64372b663e9f3ccc137
+ e78cd169d9423dc9a9126d51fbeeb91e
+ febb50152f339e23040815432e8fd6d4
+ a27b2dd3efe6a362d135d55eca70d134
+ 941ab65d0ee6a8c0578e87b8d881ffc0
+ 3bc698cc3b8937609e715c12fb135ca0
+ 8a646c02ac858485f683301e5d98adb5
+ f4656701178b9d1eea30aa73b6a96ea2
+ 11ef72971dce2d4b47f9899fabc4e591
+ aa1c16c7731e99ebc6cd536ce26daa14
+ 5dacabf98fe85149246cd5fc9bae6c39
+ 1ddf68bca0c4a97d263a2d1718f87d2e
+ e8e04eb1d9bd25e270944775d8cb4100
+ 90f738b3a6362f787e2170cb3d7317f0
+ bcf549f662ba17ac5896ef38b824f13e
+ 608118d1e8019c92a22c275b8ce02c4a
+ 5a2683f3d5a170495dc0ed3ed8cfa3f2
+ 0f64845ba2b2a4bc6a5f5637d261b93b
+ 5f873b418694a01d049664ae58ad69a8
+ cdc9eeae48d8760204735cf3dfdfbec9
+ 14d705864d455ec73ca9c373f6d61d58
+ f978f4a979edc6b3ac3c6b0cbd631c83
+ b777fc0d1a2c85ce6c0a03dfd90df48b
+ ef141db8c2469b1421c8e74f018ee8a4
+ 7a0a5d5f93e34dd40fcdce3f06848908
+ ed3685660909e7e831d7711a2cc7b7b0
+ 59341e0675583a826c34c625ba277345
+ 67a0cbf1e934dd6552cf3f231ed65964
+ 0cfed1923e5d2b849760df5ced30a175
+ 0650d1a62cce431fa07e64bf8845a904
+ 91a6f9e779d641b451efa777b6f81e81
+ ebdd6f04edccd2205bcaed1e845b567d
+ ee7f03a89cabb1370183f8a88dad1e9d
+ f88670ae7d834818ba14d71a7e0036e3
+ 7ad891738873394a7626882fd86e2031
+ be9047b6412830adcf9a7d68cb4574d1
+ b5e96c226c4a2277b34dee97efc1d960
+ cb160921e2c703bbbdafcb51d49c266f
+ f72439b0db6a44fde139cb7651c28207
+ 96aae67efd76fb6bfce1a97b1fbdbefd
+ 151fbc5f85a7acd78039d2f074901028
+ 4ab8429c7200f96c144439fe50cda12d
+ 28519bdf05d7b1c5601ebfe9b7d95c0d
+ fbee5f8f188262cf1e09c168dd1014db
+ 3031820f648fe2fb15f8c9d64c104387
+ 947daecf702214b7708c8d7df872366b
+ 14670bbba47068cc5914c399c51a20e3
+ 1d42f9adf23bf70ef1f9ba1695b9f794
+ 75204b0421e9835bf3ac3d2a0d198163
+ 6fbd208db0830b462fe2d105d1924f1e
+ 5ad79d5b9a17f3ac79b71722547fb6ff
+ 421b4ec83fe86c82e0bb952d3ee08780
+ 2d3c15a8151e86b497900f6873b1875b
+ b80a9a8c89b05c4fa22a314ea56b32ab
+ 93a573bcf722ce69de872674b3f9dd27
+ 97c2da3770e378b72637b09c0475156f
+ 572acc2315b077411699eba35c0655d1
+ 41347e33250b355077aba64c671a44ce
+ 53cd0af0a81de17378a7d3201c31dbb9
+ 99485aa18c25c0e6537334d63ccfe2c6
+ b00db640f9f0f490b0bfa7d6d8b40163
+ 04e530018ec557294dd90128fc0c4a56
+ 99539707fa1c283b6513be48996d8d14
+ 8da4141b5716e4321d1d5dcf3cfd30e6
+ 6e5a8b2abc701632f9b9ab7a839eee4d
+ 9add8ac0325547a52fa53e03b06d23b9
+ 1978498650f9f5ae335cfd702a6d16f6
+ 6952026a71d24cf0b11baca0320ab087
+ 90dc0d1bce4c19822312f51be82e9c98
+ b6959644da3e1acbdd2f63af0cc529f0
+ 5c21bdb968a38b57b911aa998b9db99d
+ 1a61bc6d32a3ed512b51535c46dc6991
+ be65efca7b7d7fc246c373e391996978
+ e772e7fae851948c6f6b19c0191821aa
+ 3de79c61240e8e04a24d95c707a61a2b
+ 9960374b772e64e5ad75e0d3b6e12f08
+ 3dfa621d920cecce3585b0f1ccb6be72
+ 687309052c27966bd4d3bbd023237c17
+ 0d615e9e715d3d4e150e836fbf43f426
+ 39b9d8b0a4aafa4e1b25992520deb011
+ ed7ad8c54b6c2d9d1727e85a1a198a31
+ 3b989b232dd121eff6b0b230e9e85f6c
+ a566bad5dc6de5bc8d782c4f292e34ed
+ 9bb9d2c8983b99f4f74e063f3d1e3479
+ d749c64a1fa96fbb0f13fc3d41b646e9
+ c6b9271b94abdf0770612a5cf4555f91
+ e1d83cafee3b3cecfdc3e473913df264
+ 9fe6b04de733a20290141310f02abdcf
+ 7bf582dfc110cfe38d616b1cdeed91ee
+ faa0e38bdadaadb61fa28e968f8a2122
+ dc188baadc7e6b3033845cfd48a14163
+ 9f814e7ec8180ca71bbaf721d9337e70
+ a969aedf03beddd8ee89367d6179e1a4
+ 15eac7459f9aeafabdce72df42a4ba69
+ 280758b9c9b1ec90f2010f1b1ab06ba7
+ ad879791f1e9d0e2d719d6a4518bbe2c
+ 4507e58972e6bc69ba4f46ea3db455ec
+ 02e681d40a2427610dd55c643e89b01d
+ 701b0401fb152e87a40c8b5babe87798
+ b177adbf225feca2f7d39f80d5acc352
+ a862bcf397791708ef2e0441282b2b69
+ c6f4dc65ba8177e50ef5742b3ecb52b7
+ 6df28cb4a7d5e8907391c3ed8e70b6e0
+ 5fee1165f23a11eeca4d24d766117377
+ 7ee82c752779d1c51f7b92660006aad2
+ 23a394d7e36aa653bff9b32fc9548250
+ e4ec374be03638c10a88f351e8f09bad
+ 26e51eabb21a9a71f1b5a685e3061843
+ 49259a0e7725bb2c6809e1aae15e8fac
+ 40801a1693b0b8bd33178d4a5bf67765
+ 7b0c70f7b8de3e1a5c19d32c36fa7d00
+ 74faff4f12efb353ccc61689fe96c46c
+ 43c98dfe30fecc4c2448dc51750154ee
+ ea0a540d06226687e3409c1226a1cc1d
+ ab8760c0219835fe97dcbbf698b39fa4
+ cc9b6238b260d924338d2c4ee2dafc95
+ 7caa4abbdb5dba23edbdc02cf2693909
+ da2990b33cef4afe7d629530b274632a
+ 65a95dd87096e2377dff6fe917befb8a
+ 02f4a01e218da00e8969d4ae52c84193
+ 2b7fbfc1efdf6ab2663e8b3d7cd3daed
+ 79da74f4e265f5290b4cf35b740663a7
+ 6531e09545716af63e4ef5cf8feb8ec1
+ 2f86aa5353798f3a4202e403bfa12c02
+ b6e5e2bb9985effc63bdcb2177ec681b
+ 937f55e89ec3df58c2d0339f9a1dd622
+ 70352371bec5bee2af6bfa7fceb7da5e
+ a53d16394c2492eb0a713ecfd8dbaf8e
+ b4fbd66018b94d83d11e3e6e7e0fac1d
+ 7c90665c0c85a3e81b15a0ea69befeeb
+ 6eea7be83224abc9a1c3da0ce2fe6c82
+ e2e0f33e2d38ad747f5c738230439ad6
+ 66de4194f057e4a5248d19c5d51f88ca
+ 6944e2533d7ddca2752df6300a281b16
+ 8e89696bf062126e4fe969a65ea5da1f
+ 466c8c681f2c6c942668d769e79defba
+ f7f176d0ff24d09134c43f0e0bc57d5b
+ 601545618b8fe3c210b4514a25242faf
+ 427efb1a07d99ca6c99698de322b7d74
+ 3926f7b7e584ad63327a622dbac60564
+ 46e6876c63888a593dd46307b7fe0965
+ 228dab05bac8112a51265a509d28adc3
+ aa747cdb68e4ce5bcf3fcf3a7fa9b371
+ fec9ddaa76f650624821341e07c32ef5
+ 8fa3ff17dbf4f5216d2b038af30b6d4b
+ 9e12e7ae2a644e77749efb5d5c4d0917
+ 7c66503ed0253f16ad021be72e4abfa7
+ 4db8a423cab299263b5139e0b44c4884
+ 267a979bb7fcadac5698365c158f8fbf
+ 14d8af5d3bc22ac49c5bdfcb8f63c9d8
+ 7afb876e7a2e7b7bfaa8619bf0c07265
+ 640eafc8068633c0e48ed3a569d18870
+ 27238520cd1123eb6e832489ed6d5a65
+ e64d386ede9f5d1ab93bbe212246086f
+ dee77f1d0d3de7f0ce6859006e0ba642
+ 93e380e1b639a5d063be34521ef058a1
+ fa4c3f5a64725039c6ac69dc817d0861
+ d9c356d0f282204a451c7acb86a0df95
+ cd6232d913be2a512d2e8fcae806afad
+ a6cef52c80b2fd0ebc43e26183d9a668
+ 792aa8e491c51b10898600e5fce883fd
+ 21e9817040f8a9ab9160745591915f28
+ a769ea7ad74deb9a793966b98be001a0
+ 664ad64136e6958a39b292ec54607037
+ 86308422b554356c850dffb9d0c90b2a
+ 96a0604d6e963d36a35e919f3ca01310
+ 868a7faef6fb3bca64a0bcb19b32840c
+ 4e24a9c27d36afaec32256a211bef2c3
+ 2771caeaa310c1a6970e95c0ea403f3a
+ dcab556a830e493703640fd1fdad70eb
+ 679c00028ac62b9aa3368840147170b5
+ 3856921a90f947ec2f3e70fa23a1f6ea
+ 9abb2fc4e373ab4431920fdaf45d8a87
+ d9b9c033f30335a81558a201810f862e
+ 0a7450c02b7ff3de085087668ba975e7
+ 712723a74eaa5f47d905c5938d1ecfa3
+ dad894bca78456b5e2ba2e2005d2f29a
+ 626844492e56226455fa3a399f41c9e9
+ e9eb491ec0bf4d3ad2c2690b94a9be95
+ 37b95c63434f398743ebbdfd2bee8089
+ 7792f06b71703a39936f75eb1f0f1e05
+ d082a2f287fc96ade10753e47cff2977
+ 39c46f24941da46177122a83f119e8b7
+ f250ae8d2a2b28eb4f736aa28e8f6abf
+ ca055bf7c4d39b07a40bb465ad635146
+ 690b3c6a74770738d2d5173fefb48e61
+ cbecc0a364fbe6a7aa0c0ea683ce62ea
+ 29a5495e10aa160cf94b0b9a06e94b48
+ 54ed408f46a6ae86d2b609c58caa2348
+ 078d68204b7b7d11103f97d6067693d2
+ 791b0a572eba95a6d0fcfdee605b284e
+ 7c5d4939a29bbde445fa5a51a806a600
+ f469bf3d05e34508c4644c9f21adde9b
+ 987d5891ea899b92697b3dbae69f4290
+ fc53b9f9e774bf455099a41e44f163e0
+ f0f2e74ef26520d16f11c26adcfa7244
+ 57853593a2c72148f0de00faa70d67d4
+ ea86ee06b4d358790036fe22ae8673a3
+ 92a9c71fb1666816273b3283e7d02504
+ 86c50734cf3e657a4bb0646ae4848af3
+ b59d6790040c27853338b680067edc52
+ 92158d7e55f4a892b3d3640cf0b254f0
+ 8518cfd8bec8278d0681f67dc7293906
+ 2e45346a0f377bcb6adc59be3c05d72a
+ 7607a3732fee1ad3b43d44875764afe0
+ 1039def0bf9d28ef2cc3a4b5649a9bcb
+ 3118e96ceee1f63c4c5f916e48a62e3c
+ 45daae280d68e757730051ce96933435
+ d27cfef60c88f2785585db762aff50c0
+ eefbe319cbc34fed4d63ace54f2fc1c8
+ 806acc4c1616d1d940e61d27737e39f6
+ f6ce2c5a24ea1545e1d22956017e31b0
+ 66cee3c0d3e34c4bd97599c1a633c7db
+ 71c35b11bb1662bd612bc8e3e21232e5
+ 50cd266f7a98b7d19ec75b9be804e715
+ e5fd32dabb3412be5797805b5b00c002
+ 7a7bd91ea1d3c40e3f52a3453f6d5789
+ 0c2692680244c3848e16c5f5cc6e525f
+ 3126bd7f0dc24fb15fbaa6516bbc08d9
+ 1b1eb3803d01ea5dbfaee1836c9efb02
+ 85770badb88a7a876bb828718cc4a30f
+ 1c9f328d29ee7e2453bf30a6b05939f4
+ 5d6709607fc95b3867fc491c0fe5e5e3
+ ee0b3f5284765e02f1357f1cf7071ee1
+ 39f6e98033e45e3b599cdff5b72d7328
+ 9dc57923ec3e984723338db44460b4fe
+ f2eabbfcc31b0ab183fb4b91ba4f19c4
+ cce4ae63928058699f87b61a9604d1b1
+ 04d855def138064915a06781720d33cc
+ a4e382d1493412f3d5cb1e06f9ec9b88
+ c209d368e99561002a43697c0496d87d
+ 749a85a21354df350ddb8c50a5561b22
+ 83cebd2f3cf663215a8bb97dcd38b13e
+ b867555c6e06befe0715665c2a10931d
+ e514c7719ac869a86ff466a22de3554d
+ c1e1b2a685c99e73720bca83f0c01405
+ 2db3c206f581c88c329567013d16aca3
+ 62ff0d85b3b5d71c2854623bcad155cb
+ 44106df1f8f1145bb7f9d465a3d85233
+ bf82709deb10a369d7eaa23d623f631a
+ f32ab303e103ceeb6ca46926f2c1f510
+ d6d11bb00006f7b6da2f4939ef06af23
+ 01893b6c69d97c42f223d95ff8595ff7
+ ec52deb7d44c5186c324ac0532ff2bc8
+ 60ffdd2bd30dc4a02dfca2ced49bbec5
+ 4a8d5a29427d179a6b0566d79555cbad
+ eedc5b614bac903c03dfdd287dc0123a
+ e1f2b5c9d0a65366efd6f4840725d3c5
+ e307c8d98edfad2432ceea160a5b455d
+ 3c40924714e6511a2c7c2e8d64a22e4c
+ e9c38c20924902a8cb168d69adf028c5
+ e01bdd83f13783e98eae4d89d7c55405
+
diff --git a/catalog/fgaddon-catalog/template.xml b/catalog/fgaddon-catalog/template.xml
new file mode 100644
index 0000000..b3b5275
--- /dev/null
+++ b/catalog/fgaddon-catalog/template.xml
@@ -0,0 +1,23 @@
+
+
+
+
+ 3.4.*
+ 3.5.*
+ 3.6.*
+ 3.7.*
+ 2016.1.*
+ 2016.*.*
+ org.flightgear.fgaddon
+ GPL
+ http://mirrors.ibiblio.org/flightgear/ftp/Aircraft/catalog.xml
+ FlightGear Aircraft Distribution From fgaddon
+ This hangar provides aircraft officially supported and maintained by the FlightGear project, under a free-software license.
+
+ Auf Deutsch
+
+
+ En Francais
+
+
+
diff --git a/catalog/fgaddon-catalog/zip-excludes.lst b/catalog/fgaddon-catalog/zip-excludes.lst
new file mode 100644
index 0000000..9f4ab72
--- /dev/null
+++ b/catalog/fgaddon-catalog/zip-excludes.lst
@@ -0,0 +1,5 @@
+*/.svn/*
+*.xcf
+*.blend
+*.psd
+*~
diff --git a/catalog/update-catalog.py b/catalog/update-catalog.py
new file mode 100755
index 0000000..0177b8c
--- /dev/null
+++ b/catalog/update-catalog.py
@@ -0,0 +1,332 @@
+#!/usr/bin/python
+
+import argparse
+import datetime
+import hashlib # md5
+import lxml.etree as ET
+import os
+import re
+import shutil
+import subprocess
+import time
+
+CATALOG_VERSION = 4
+
+parser = argparse.ArgumentParser()
+parser.add_argument("--update", help="Update/pull SCM source",
+ action="store_true")
+parser.add_argument("--no-update",
+ help="Disable updating from SCM source",
+ action="store_true")
+parser.add_argument("--clean", help="Force regeneration of all zip files",
+ action="store_true")
+parser.add_argument("dir", help="Catalog directory")
+args = parser.parse_args()
+
+# xml node (robust) get text helper
+def get_xml_text(e):
+ if e != None and e.text != None:
+ return e.text
+ else:
+ return ''
+
+# create an xml node with text content
+def make_xml_leaf(name, text):
+ leaf = ET.Element(name)
+ if text != None:
+ if isinstance(text, (int, long)):
+ leaf.text = str(text)
+ else:
+ leaf.text = text
+ else:
+ leaf.text = ''
+ return leaf
+
+# return all available aircraft information from the set file as a
+# dict
+def scan_set_file(set_file):
+ base_file = os.path.basename(set_file)
+ base_id = base_file[:-8]
+ #print ' scanning:', base_file
+ parser = ET.XMLParser(remove_blank_text=True)
+ set_xml = ET.parse(set_file, parser)
+ set_node = set_xml.getroot()
+ sim_node = set_node.find('sim')
+ if sim_node == None:
+ return None
+ variant = {}
+ variant['name'] = get_xml_text(sim_node.find('description'))
+ variant['status'] = get_xml_text(sim_node.find('status'))
+ variant['author'] = get_xml_text(sim_node.find('author'))
+ variant['description'] = get_xml_text(sim_node.find('long-description'))
+ variant['id'] = base_id
+ rating_node = sim_node.find('rating')
+ if rating_node != None:
+ variant['rating_FDM'] = int(get_xml_text(rating_node.find('FDM')))
+ variant['rating_systems'] = int(get_xml_text(rating_node.find('systems')))
+ variant['rating_cockpit'] = int(get_xml_text(rating_node.find('cockpit')))
+ variant['rating_model'] = int(get_xml_text(rating_node.find('model')))
+ variant['variant-of'] = get_xml_text(sim_node.find('variant-of'))
+ #print ' ', variant
+ return variant
+
+# scan all the -set.xml files in an aircraft directory. Returns a
+# package dict and a list of variants.
+def scan_aircraft_dir(aircraft_dir):
+ found_master = False
+ package = None
+ variants = []
+ files = os.listdir(aircraft_dir)
+ for file in sorted(files, key=lambda s: s.lower()):
+ if file.endswith('-set.xml'):
+ variant = scan_set_file(os.path.join(aircraft_dir, file))
+ if variant == None:
+ continue
+ if package == None:
+ # just in case no one claims to be master, the first
+ # variant defaults to master, but we will overwrite
+ # this if we find something better.
+ package = variant
+ if not found_master and variant['variant-of'] == '':
+ found_master = True
+ package = variant
+ else:
+ variants.append( {'id': variant['id'],
+ 'name': variant['name'] } )
+ return (package, variants)
+
+# use svn commands to report the last change date within dir
+def last_change_date_svn(dir):
+ command = [ 'svn', 'info', dir ]
+ result = subprocess.check_output( command )
+ match = re.search('Last Changed Date: (\d+)\-(\d+)\-(\d+)', result)
+ if match:
+ rev_str = match.group(1) + match.group(2) + match.group(3)
+ return int(rev_str)
+
+# find the most recent mtime within a directory subtree
+def scan_dir_for_change_date_mtime(path):
+ maxsec = 0
+ names = os.listdir(path)
+ for name in names:
+ fullname = os.path.join(path, name)
+ if name == '.' or name == '..':
+ pass
+ elif os.path.isdir( fullname ):
+ mtime = scan_dir_for_change_date_mtime( fullname )
+ if mtime > maxsec:
+ maxsec = mtime
+ else:
+ mtime = os.path.getmtime( fullname )
+ if mtime > maxsec:
+ maxsec = mtime
+ return maxsec
+
+def make_aircraft_zip(repo_path, name, zip_file):
+ print "Updating:", name + '.zip'
+ os.chdir(repo_path)
+ if os.path.exists(zip_file):
+ os.remove(zip_file)
+ command = ['zip', '-rq', '-9']
+ if os.path.exists(zip_excludes):
+ command += ['-x@' + zip_excludes]
+ else:
+ print "warning: no zip-excludes.lst file provided", zip_excludes
+ command += [zip_file, name]
+ subprocess.call(command)
+
+def get_md5sum(file):
+ f = open(file, 'r')
+ md5sum = hashlib.md5(f.read()).hexdigest()
+ return md5sum
+
+#def get_file_stats(file):
+# f = open(file, 'r')
+# md5 = hashlib.md5(f.read()).hexdigest()
+# file_size = os.path.getsize(file)
+# return (md5, file_size)
+
+if not os.path.isdir(args.dir):
+ print "A valid catalog directory must be provided"
+ exit(0)
+
+parser = ET.XMLParser(remove_blank_text=True)
+
+config_file = os.path.join(args.dir, 'catalog.config.xml')
+config = ET.parse(config_file, parser)
+config_node = config.getroot()
+
+template_file = os.path.join(args.dir, 'template.xml')
+template = ET.parse(template_file, parser)
+template_root = template.getroot()
+template_node = template_root.find('template')
+
+md5sum_file = os.path.join(args.dir, 'md5sum.xml')
+if os.path.exists(md5sum_file):
+ md5sum_tree = ET.parse(md5sum_file, parser)
+ md5sum_root = md5sum_tree.getroot()
+else:
+ md5sum_root = ET.Element('PropertyList')
+ md5sum_tree = ET.ElementTree(md5sum_root)
+
+scm_list = config_node.findall('scm')
+upload_node = config_node.find('upload')
+download_base = get_xml_text(config_node.find('download-url'))
+output_dir = get_xml_text(config_node.find('local-output'))
+if output_dir == '':
+ output_dir = os.path.join(args.dir, 'output')
+if not os.path.isdir(output_dir):
+ os.mkdir(output_dir)
+thumbnail_dir = os.path.join(output_dir, 'thumbnails')
+if not os.path.isdir(thumbnail_dir):
+ os.mkdir(thumbnail_dir)
+tmp = os.path.join(args.dir, 'zip-excludes.lst')
+zip_excludes = os.path.realpath(tmp)
+
+# freshen repositories
+if args.no_update:
+ print 'Skipping repository updates.'
+else:
+ cwd = os.getcwd()
+ for scm in scm_list:
+ repo_type = get_xml_text(scm.find('type'))
+ repo_path = get_xml_text(scm.find('path'))
+ if repo_type == 'svn':
+ print 'SVN update:', repo_path
+ subprocess.call(['svn', 'update', repo_path])
+ elif repo_type == 'git':
+ print 'GIT pull:', repo_path
+ os.chdir(repo_path)
+ subprocess.call(['git','pull'])
+ elif repo_type == 'no-scm':
+ print "No update of unmannaged files:", repo_path
+ else:
+ print "Unknown scm type:", scm, repo_path
+ os.chdir(cwd)
+
+# names of zip files we want (so we can identify/remove orphans)
+valid_zips = []
+
+# create the catalog tree
+catalog_node = ET.Element('PropertyList')
+catalog_root = ET.ElementTree(catalog_node)
+
+# include the template configuration
+for child in template_node:
+ catalog_node.append(child)
+
+# scan repositories for catalog information
+for scm in scm_list:
+ repo_type = get_xml_text(scm.find('type'))
+ repo_path = get_xml_text(scm.find('path'))
+ skip_nodes = scm.findall('skip')
+ skip_list = []
+ for s in skip_nodes:
+ skip_list.append(get_xml_text(s))
+ print 'skip list:', skip_list
+ names = os.listdir(repo_path)
+ for name in sorted(names, key=lambda s: s.lower()):
+ if name in skip_list:
+ print "skipping:", name
+ continue
+ aircraft_dir = os.path.join(repo_path, name)
+ if os.path.isdir(aircraft_dir):
+ print "%s:" % name,
+ (package, variants) = scan_aircraft_dir(aircraft_dir)
+ if package == None:
+ print "skipping:", name, "(no -set.xml files)"
+ continue
+ #print "package:", package
+ #print "variants:", variants
+ package_node = ET.Element('package')
+ package_node.append( make_xml_leaf('name', package['name']) )
+ package_node.append( make_xml_leaf('status', package['status']) )
+ package_node.append( make_xml_leaf('author', package['author']) )
+ package_node.append( make_xml_leaf('description', package['description']) )
+ if 'rating_FDM' in package or 'rating_systems' in package \
+ or 'rating_cockpit' in package or 'rating_model' in package:
+ rating_node = ET.Element('rating')
+ package_node.append(rating_node)
+ rating_node.append( make_xml_leaf('FDM',
+ package['rating_FDM']) )
+ rating_node.append( make_xml_leaf('systems',
+ package['rating_systems']) )
+ rating_node.append( make_xml_leaf('cockpit',
+ package['rating_cockpit']) )
+ rating_node.append( make_xml_leaf('model',
+ package['rating_model']) )
+ package_node.append( make_xml_leaf('id', package['id']) )
+ for variant in variants:
+ variant_node = ET.Element('variant')
+ package_node.append(variant_node)
+ variant_node.append( make_xml_leaf('id', variant['id']) )
+ variant_node.append( make_xml_leaf('name', variant['name']) )
+ package_node.append( make_xml_leaf('dir', name) )
+ if not download_base.endswith('/'):
+ download_base += '/'
+ download_url = download_base + name + '.zip'
+ thumbnail_url = download_base + 'thumbnails/' + name + '_thumbnail.jpg'
+ package_node.append( make_xml_leaf('url', download_url) )
+ package_node.append( make_xml_leaf('thumbnail', thumbnail_url) )
+
+ # todo: url (download), thumbnail (download url)
+
+ # get cached md5sum if it exists
+ md5sum = get_xml_text(md5sum_root.find(str('aircraft_' + name)))
+
+ # now do the packaging and rev number stuff
+ dir_mtime = scan_dir_for_change_date_mtime(aircraft_dir)
+ if repo_type == 'svn':
+ rev = last_change_date_svn(aircraft_dir)
+ else:
+ d = datetime.datetime.utcfromtimestamp(dir_mtime)
+ rev = d.strftime("%Y%m%d")
+ package_node.append( make_xml_leaf('revision', rev) )
+ #print "rev:", rev
+ #print "dir mtime:", dir_mtime
+ zipfile = os.path.join( output_dir, name + '.zip' )
+ valid_zips.append(name + '.zip')
+ if not os.path.exists(zipfile) \
+ or dir_mtime > os.path.getmtime(zipfile) \
+ or args.clean:
+ # rebuild zip file
+ print "updating:", zipfile
+ make_aircraft_zip(repo_path, name, zipfile)
+ md5sum = get_md5sum(zipfile)
+ else:
+ print "(no change)"
+ if md5sum == "":
+ md5sum = get_md5sum(zipfile)
+ filesize = os.path.getsize(zipfile)
+ package_node.append( make_xml_leaf('md5', md5sum) )
+ package_node.append( make_xml_leaf('file-size-bytes', filesize) )
+
+ # handle md5sum cache
+ node = md5sum_root.find('aircraft_' + name)
+ if node != None:
+ node.text = md5sum
+ else:
+ md5sum_root.append( make_xml_leaf('aircraft_' + name, md5sum) )
+
+ # handle thumbnails
+ thumbnail_src = os.path.join(aircraft_dir, 'thumbnail.jpg')
+ thumbnail_dst = os.path.join(thumbnail_dir, name + '_thumbnail.jpg')
+ if os.path.exists(thumbnail_src):
+ shutil.copy2(thumbnail_src, thumbnail_dst)
+ catalog_node.append(package_node)
+ package_node.append( make_xml_leaf('thumbnail-path', 'thumbnail.jpg') )
+
+# write out the master catalog file
+cat_file = os.path.join(output_dir, 'catalog.xml')
+catalog_root.write(cat_file, encoding='utf-8', xml_declaration=True, pretty_print=True)
+
+# write out the md5sum cache file
+print md5sum_file
+md5sum_tree.write(md5sum_file, encoding='utf-8', xml_declaration=True, pretty_print=True)
+
+# look for orphaned zip files
+files = os.listdir(output_dir)
+for file in files:
+ if file.endswith('.zip')and not file in valid_zips:
+ print "orphaned zip:", file
+