ApiController.php 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. <?php
  2. /*
  3. * This file is part of Packagist.
  4. *
  5. * (c) Jordi Boggiano <j.boggiano@seld.be>
  6. * Nils Adermann <naderman@naderman.de>
  7. *
  8. * For the full copyright and license information, please view the LICENSE
  9. * file that was distributed with this source code.
  10. */
  11. namespace Packagist\WebBundle\Controller;
  12. use Composer\IO\BufferIO;
  13. use Composer\Factory;
  14. use Composer\Repository\VcsRepository;
  15. use Composer\Package\Loader\ValidatingArrayLoader;
  16. use Composer\Package\Loader\ArrayLoader;
  17. use Packagist\WebBundle\Package\Updater;
  18. use Packagist\WebBundle\Entity\Package;
  19. use Symfony\Component\HttpFoundation\Response;
  20. use Symfony\Component\HttpFoundation\Request;
  21. use Symfony\Component\Console\Output\OutputInterface;
  22. use Sensio\Bundle\FrameworkExtraBundle\Configuration\Template;
  23. use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route;
  24. use Sensio\Bundle\FrameworkExtraBundle\Configuration\Method;
  25. /**
  26. * @author Jordi Boggiano <j.boggiano@seld.be>
  27. */
  28. class ApiController extends Controller
  29. {
  30. /**
  31. * @Template()
  32. * @Route("/packages.json", name="packages", defaults={"_format" = "json"})
  33. */
  34. public function packagesAction()
  35. {
  36. // fallback if any of the dumped files exist
  37. $rootJson = $this->container->getParameter('kernel.root_dir').'/../web/packages_root.json';
  38. if (file_exists($rootJson)) {
  39. return new Response(file_get_contents($rootJson));
  40. }
  41. $rootJson = $this->container->getParameter('kernel.root_dir').'/../web/packages.json';
  42. if (file_exists($rootJson)) {
  43. return new Response(file_get_contents($rootJson));
  44. }
  45. $em = $this->get('doctrine')->getEntityManager();
  46. gc_enable();
  47. $packages = $em->getRepository('Packagist\WebBundle\Entity\Package')
  48. ->getFullPackages();
  49. $notifyUrl = $this->generateUrl('track_download', array('name' => 'VND/PKG'));
  50. $data = array(
  51. 'notify' => str_replace('VND/PKG', '%package%', $notifyUrl),
  52. 'packages' => array(),
  53. );
  54. foreach ($packages as $package) {
  55. $versions = array();
  56. foreach ($package->getVersions() as $version) {
  57. $versions[$version->getVersion()] = $version->toArray();
  58. $em->detach($version);
  59. }
  60. $data['packages'][$package->getName()] = $versions;
  61. $em->detach($package);
  62. }
  63. unset($versions, $package, $packages);
  64. $response = new Response(json_encode($data), 200);
  65. $response->setSharedMaxAge(120);
  66. return $response;
  67. }
  68. /**
  69. * @Route("/api/github", name="github_postreceive", defaults={"_format" = "json"})
  70. * @Method({"POST"})
  71. */
  72. public function githubPostReceive(Request $request)
  73. {
  74. return $this->receivePost($request, '{^(?:https?://|git://|git@)?(?P<domain>github\.com)[:/](?P<repo>[\w.-]+/[\w.-]+?)(?:\.git)?$}');
  75. }
  76. /**
  77. * @Route("/api/bitbucket", name="bitbucket_postreceive", defaults={"_format" = "json"})
  78. * @Method({"POST"})
  79. */
  80. public function bitbucketPostReceive(Request $request)
  81. {
  82. return $this->receivePost($request, '{^(?:https?://)?(?P<domain>bitbucket\.org)/(?P<repo>[\w.-]+/[\w.-]+?)/?$}');
  83. }
  84. /**
  85. * @Route("/downloads/{name}", name="track_download", requirements={"name"="[A-Za-z0-9_.-]+/[A-Za-z0-9_.-]+"}, defaults={"_format" = "json"})
  86. * @Method({"POST"})
  87. */
  88. public function trackDownloadAction(Request $request, $name)
  89. {
  90. $result = $this->get('doctrine.dbal.default_connection')->fetchAssoc(
  91. 'SELECT p.id, v.id vid
  92. FROM package p
  93. LEFT JOIN package_version v ON p.id = v.package_id
  94. WHERE p.name = ?
  95. AND v.normalizedVersion = ?
  96. LIMIT 1',
  97. array($name, $request->request->get('version_normalized'))
  98. );
  99. if (!$result) {
  100. return new Response('{"status": "error", "message": "Package not found"}', 200);
  101. }
  102. $redis = $this->get('snc_redis.default');
  103. $id = $result['id'];
  104. $version = $result['vid'];
  105. $throttleKey = 'dl:'.$id.':'.$request->getClientIp().':'.date('Ymd');
  106. $requests = $redis->incr($throttleKey);
  107. if (1 === $requests) {
  108. $redis->expire($throttleKey, 86400);
  109. }
  110. if ($requests <= 10) {
  111. $redis->incr('downloads');
  112. $redis->incr('dl:'.$id);
  113. $redis->incr('dl:'.$id.':'.date('Ym'));
  114. $redis->incr('dl:'.$id.':'.date('Ymd'));
  115. $redis->incr('dl:'.$id.'-'.$version);
  116. $redis->incr('dl:'.$id.'-'.$version.':'.date('Ym'));
  117. $redis->incr('dl:'.$id.'-'.$version.':'.date('Ymd'));
  118. }
  119. return new Response('{"status": "success"}', 201);
  120. }
  121. protected function receivePost(Request $request, $urlRegex)
  122. {
  123. $payload = json_decode($request->request->get('payload'), true);
  124. if (!$payload || !isset($payload['repository']['url'])) {
  125. return new Response(json_encode(array('status' => 'error', 'message' => 'Missing or invalid payload',)), 406);
  126. }
  127. // try to parse the URL first to avoid the DB lookup on malformed requests
  128. if (!preg_match($urlRegex, $payload['repository']['url'], $requestedRepo)) {
  129. return new Response(json_encode(array('status' => 'error', 'message' => 'Could not parse payload repository URL',)), 406);
  130. }
  131. $username = $request->request->has('username') ?
  132. $request->request->get('username') :
  133. $request->query->get('username');
  134. $apiToken = $request->request->has('apiToken') ?
  135. $request->request->get('apiToken') :
  136. $request->query->get('apiToken');
  137. $user = $this->get('packagist.user_repository')
  138. ->findOneBy(array('username' => $username, 'apiToken' => $apiToken));
  139. if (!$user) {
  140. return new Response(json_encode(array('status' => 'error', 'message' => 'Invalid credentials',)), 403);
  141. }
  142. $updated = false;
  143. $config = Factory::createConfig();
  144. $loader = new ValidatingArrayLoader(new ArrayLoader());
  145. $updater = $this->get('packagist.package_updater');
  146. $em = $this->get('doctrine.orm.entity_manager');
  147. $candidate = array();
  148. foreach ($user->getPackages() as $package) {
  149. if (preg_match($urlRegex, $package->getRepository(), $candidate)
  150. && $candidate['domain'] === $requestedRepo['domain']
  151. && $candidate['repo'] === $requestedRepo['repo']
  152. ) {
  153. set_time_limit(3600);
  154. $updated = true;
  155. $io = new BufferIO('', OutputInterface::VERBOSITY_VERBOSE);
  156. $repository = new VcsRepository(array('url' => $package->getRepository()), $io, $config);
  157. $repository->setLoader($loader);
  158. $package->setAutoUpdated(true);
  159. $em->flush();
  160. try {
  161. $updater->update($package, $repository);
  162. if ($repository->hadInvalidBranches()) {
  163. throw new \RuntimeException('Some branches contained invalid data and were discarded, it is advised to review the log and fix any issues present in branches');
  164. }
  165. } catch (\Exception $e) {
  166. // TODO send email to maintainer(s)
  167. return new Response(json_encode(array(
  168. 'status' => 'error',
  169. 'message' => '['.get_class($e).'] '.$e->getMessage(),
  170. 'details' => '<pre>'.$io->getOutput().'</pre>'
  171. )), 400);
  172. }
  173. }
  174. }
  175. if ($updated) {
  176. return new Response('{"status": "success"}', 202);
  177. }
  178. return new Response(json_encode(array('status' => 'error', 'message' => 'Could not find a package that matches this request (does user maintain the package?)',)), 404);
  179. }
  180. }