ApiController.php 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  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\NullIO;
  13. use Composer\Repository\VcsRepository;
  14. use Packagist\WebBundle\Package\Updater;
  15. use Symfony\Bundle\FrameworkBundle\Controller\Controller;
  16. use Symfony\Component\HttpFoundation\Response;
  17. use Symfony\Component\HttpFoundation\Request;
  18. use Sensio\Bundle\FrameworkExtraBundle\Configuration\Template;
  19. use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route;
  20. use Sensio\Bundle\FrameworkExtraBundle\Configuration\Method;
  21. /**
  22. * @author Jordi Boggiano <j.boggiano@seld.be>
  23. */
  24. class ApiController extends Controller
  25. {
  26. /**
  27. * @Template()
  28. * @Route("/packages.json", name="packages", defaults={"_format" = "json"})
  29. */
  30. public function packagesAction()
  31. {
  32. $em = $this->get('doctrine')->getEntityManager();
  33. gc_enable();
  34. $packages = $em->getRepository('Packagist\WebBundle\Entity\Package')
  35. ->getFullPackages();
  36. $data = array();
  37. foreach ($packages as $package) {
  38. $versions = array();
  39. foreach ($package->getVersions() as $version) {
  40. $versions[$version->getVersion()] = $version->toArray();
  41. $em->detach($version);
  42. }
  43. $data[$package->getName()] = array('versions' => $versions);
  44. $em->detach($package);
  45. }
  46. unset($versions, $package, $packages);
  47. $response = new Response(json_encode($data), 200);
  48. $response->setSharedMaxAge(120);
  49. return $response;
  50. }
  51. /**
  52. * @Route("/api/github", name="github_postreceive", defaults={"_format" = "json"})
  53. * @Method({"POST"})
  54. */
  55. public function githubPostReceive(Request $request)
  56. {
  57. $payload = json_decode($request->request->get('payload'), true);
  58. if (!$payload || !isset($payload['repository']['url'])) {
  59. return new Response(json_encode(array('status' => 'error', 'message' => 'Missing or invalid payload',)), 406);
  60. }
  61. $username = $request->request->has('username') ?
  62. $request->request->get('username') :
  63. $request->query->get('username');
  64. $apiToken = $request->request->has('apiToken') ?
  65. $request->request->get('apiToken') :
  66. $request->query->get('apiToken');
  67. $doctrine = $this->get('doctrine');
  68. $user = $doctrine
  69. ->getRepository('PackagistWebBundle:User')
  70. ->findOneBy(array('username' => $username, 'apiToken' => $apiToken));
  71. if (!$user) {
  72. return new Response(json_encode(array('status' => 'error', 'message' => 'Invalid credentials',)), 403);
  73. }
  74. if (!preg_match('{github.com/[\w.-]+/[\w.-]+$}', $payload['repository']['url'], $match)) {
  75. return new Response(json_encode(array('status' => 'error', 'message' => 'Could not parse payload repository URL',)), 406);
  76. }
  77. $payloadRepositoryChunk = $match[0];
  78. foreach ($user->getPackages() as $package) {
  79. if (false !== strpos($package->getRepository(), $payloadRepositoryChunk)) {
  80. // We found the package that was referenced.
  81. $updater = new Updater($doctrine);
  82. $repository = new VcsRepository(array('url' => $package->getRepository()), new NullIO);
  83. $package->setAutoUpdated(true);
  84. $updater->update($package, $repository);
  85. return new Response('{"status": "success"}', 202);
  86. }
  87. }
  88. return new Response(json_encode(array('status' => 'error', 'message' => 'Could not find a package that matches this request (does user maintain the package?)',)), 404);
  89. }
  90. }