router.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552
  1. package router
  2. import (
  3. "net/http"
  4. "os"
  5. "github.com/go-chi/chi"
  6. "github.com/gorilla/sessions"
  7. "github.com/porter-dev/porter/internal/repository"
  8. "github.com/porter-dev/porter/server/api"
  9. "github.com/porter-dev/porter/server/requestlog"
  10. mw "github.com/porter-dev/porter/server/router/middleware"
  11. )
  12. // New creates a new Chi router instance
  13. func New(
  14. a *api.App,
  15. store sessions.Store,
  16. cookieName string,
  17. staticFilePath string,
  18. repo *repository.Repository,
  19. ) *chi.Mux {
  20. l := a.Logger()
  21. r := chi.NewRouter()
  22. auth := mw.NewAuth(store, cookieName, repo)
  23. r.Route("/api", func(r chi.Router) {
  24. r.Use(mw.ContentTypeJSON)
  25. // health checks
  26. r.Method("GET", "/livez", http.HandlerFunc(a.HandleLive))
  27. r.Method("GET", "/readyz", http.HandlerFunc(a.HandleReady))
  28. // /api/users routes
  29. r.Method(
  30. "GET",
  31. "/users/{user_id}",
  32. auth.DoesUserIDMatch(
  33. requestlog.NewHandler(a.HandleReadUser, l),
  34. mw.URLParam,
  35. ),
  36. )
  37. r.Method(
  38. "GET",
  39. "/users/{user_id}/projects",
  40. auth.DoesUserIDMatch(
  41. requestlog.NewHandler(a.HandleListUserProjects, l),
  42. mw.URLParam,
  43. ),
  44. )
  45. r.Method(
  46. "POST",
  47. "/users",
  48. requestlog.NewHandler(a.HandleCreateUser, l),
  49. )
  50. r.Method(
  51. "DELETE",
  52. "/users/{user_id}",
  53. auth.DoesUserIDMatch(
  54. requestlog.NewHandler(a.HandleDeleteUser, l),
  55. mw.URLParam,
  56. ),
  57. )
  58. r.Method(
  59. "POST",
  60. "/login",
  61. requestlog.NewHandler(a.HandleLoginUser, l),
  62. )
  63. r.Method(
  64. "GET",
  65. "/auth/check",
  66. auth.BasicAuthenticate(
  67. requestlog.NewHandler(a.HandleAuthCheck, l),
  68. ),
  69. )
  70. r.Method(
  71. "POST",
  72. "/logout",
  73. auth.BasicAuthenticate(
  74. requestlog.NewHandler(a.HandleLogoutUser, l),
  75. ),
  76. )
  77. // /api/integrations routes
  78. r.Method(
  79. "GET",
  80. "/integrations/cluster",
  81. auth.BasicAuthenticate(
  82. requestlog.NewHandler(a.HandleListClusterIntegrations, l),
  83. ),
  84. )
  85. r.Method(
  86. "GET",
  87. "/integrations/registry",
  88. auth.BasicAuthenticate(
  89. requestlog.NewHandler(a.HandleListRegistryIntegrations, l),
  90. ),
  91. )
  92. r.Method(
  93. "GET",
  94. "/integrations/repo",
  95. auth.BasicAuthenticate(
  96. requestlog.NewHandler(a.HandleListRepoIntegrations, l),
  97. ),
  98. )
  99. // /api/templates routes
  100. r.Method(
  101. "GET",
  102. "/templates",
  103. auth.BasicAuthenticate(
  104. requestlog.NewHandler(a.HandleListTemplates, l),
  105. ),
  106. )
  107. // /api/oauth routes
  108. // r.Method(
  109. // "GET",
  110. // "/oauth/projects/{project_id}/github",
  111. // auth.DoesUserHaveProjectAccess(
  112. // requestlog.NewHandler(a.HandleGithubOAuthStartProject, l),
  113. // mw.URLParam,
  114. // mw.WriteAccess,
  115. // ),
  116. // )
  117. // r.Method(
  118. // "GET",
  119. // "/oauth/github/callback",
  120. // requestlog.NewHandler(a.HandleGithubOAuthCallback, l),
  121. // )
  122. // /api/projects routes
  123. r.Method(
  124. "GET",
  125. "/projects/{project_id}",
  126. auth.DoesUserHaveProjectAccess(
  127. requestlog.NewHandler(a.HandleReadProject, l),
  128. mw.URLParam,
  129. mw.ReadAccess,
  130. ),
  131. )
  132. r.Method(
  133. "POST",
  134. "/projects",
  135. auth.BasicAuthenticate(
  136. requestlog.NewHandler(a.HandleCreateProject, l),
  137. ),
  138. )
  139. r.Method(
  140. "DELETE",
  141. "/projects/{project_id}",
  142. auth.DoesUserHaveProjectAccess(
  143. requestlog.NewHandler(a.HandleDeleteProject, l),
  144. mw.URLParam,
  145. mw.WriteAccess,
  146. ),
  147. )
  148. // /api/projects/{project_id}/clusters routes
  149. r.Method(
  150. "GET",
  151. "/projects/{project_id}/clusters",
  152. auth.DoesUserHaveProjectAccess(
  153. requestlog.NewHandler(a.HandleListProjectClusters, l),
  154. mw.URLParam,
  155. mw.ReadAccess,
  156. ),
  157. )
  158. r.Method(
  159. "POST",
  160. "/projects/{project_id}/clusters",
  161. auth.DoesUserHaveProjectAccess(
  162. requestlog.NewHandler(a.HandleCreateProjectCluster, l),
  163. mw.URLParam,
  164. mw.ReadAccess,
  165. ),
  166. )
  167. r.Method(
  168. "GET",
  169. "/projects/{project_id}/clusters/{cluster_id}",
  170. auth.DoesUserHaveProjectAccess(
  171. auth.DoesUserHaveClusterAccess(
  172. requestlog.NewHandler(a.HandleReadProjectCluster, l),
  173. mw.URLParam,
  174. mw.URLParam,
  175. ),
  176. mw.URLParam,
  177. mw.ReadAccess,
  178. ),
  179. )
  180. r.Method(
  181. "DELETE",
  182. "/projects/{project_id}/clusters/{cluster_id}",
  183. auth.DoesUserHaveProjectAccess(
  184. auth.DoesUserHaveClusterAccess(
  185. requestlog.NewHandler(a.HandleDeleteProjectCluster, l),
  186. mw.URLParam,
  187. mw.URLParam,
  188. ),
  189. mw.URLParam,
  190. mw.WriteAccess,
  191. ),
  192. )
  193. // /api/projects/{project_id}/clusters/candidates routes
  194. r.Method(
  195. "POST",
  196. "/projects/{project_id}/clusters/candidates",
  197. auth.DoesUserHaveProjectAccess(
  198. requestlog.NewHandler(a.HandleCreateProjectClusterCandidates, l),
  199. mw.URLParam,
  200. mw.WriteAccess,
  201. ),
  202. )
  203. r.Method(
  204. "GET",
  205. "/projects/{project_id}/clusters/candidates",
  206. auth.DoesUserHaveProjectAccess(
  207. requestlog.NewHandler(a.HandleListProjectClusterCandidates, l),
  208. mw.URLParam,
  209. mw.WriteAccess,
  210. ),
  211. )
  212. r.Method(
  213. "POST",
  214. "/projects/{project_id}/clusters/candidates/{candidate_id}/resolve",
  215. auth.DoesUserHaveProjectAccess(
  216. requestlog.NewHandler(a.HandleResolveClusterCandidate, l),
  217. mw.URLParam,
  218. mw.WriteAccess,
  219. ),
  220. )
  221. // /api/projects/{project_id}/integrations routes
  222. r.Method(
  223. "POST",
  224. "/projects/{project_id}/integrations/gcp",
  225. auth.DoesUserHaveProjectAccess(
  226. requestlog.NewHandler(a.HandleCreateGCPIntegration, l),
  227. mw.URLParam,
  228. mw.WriteAccess,
  229. ),
  230. )
  231. r.Method(
  232. "POST",
  233. "/projects/{project_id}/integrations/aws",
  234. auth.DoesUserHaveProjectAccess(
  235. requestlog.NewHandler(a.HandleCreateAWSIntegration, l),
  236. mw.URLParam,
  237. mw.WriteAccess,
  238. ),
  239. )
  240. // /api/projects/{project_id}/registries routes
  241. r.Method(
  242. "POST",
  243. "/projects/{project_id}/registries",
  244. auth.DoesUserHaveProjectAccess(
  245. requestlog.NewHandler(a.HandleCreateRegistry, l),
  246. mw.URLParam,
  247. mw.WriteAccess,
  248. ),
  249. )
  250. r.Method(
  251. "GET",
  252. "/projects/{project_id}/registries",
  253. auth.DoesUserHaveProjectAccess(
  254. requestlog.NewHandler(a.HandleListProjectRegistries, l),
  255. mw.URLParam,
  256. mw.WriteAccess,
  257. ),
  258. )
  259. r.Method(
  260. "DELETE",
  261. "/projects/{project_id}/registries/{registry_id}",
  262. auth.DoesUserHaveProjectAccess(
  263. auth.DoesUserHaveRegistryAccess(
  264. requestlog.NewHandler(a.HandleDeleteProjectRegistry, l),
  265. mw.URLParam,
  266. mw.URLParam,
  267. ),
  268. mw.URLParam,
  269. mw.WriteAccess,
  270. ),
  271. )
  272. // /api/projects/{project_id}/registries/{registry_id}/repositories routes
  273. r.Method(
  274. "GET",
  275. "/projects/{project_id}/registries/{registry_id}/repositories",
  276. auth.DoesUserHaveProjectAccess(
  277. auth.DoesUserHaveRegistryAccess(
  278. requestlog.NewHandler(a.HandleListRepositories, l),
  279. mw.URLParam,
  280. mw.URLParam,
  281. ),
  282. mw.URLParam,
  283. mw.WriteAccess,
  284. ),
  285. )
  286. r.Method(
  287. "GET",
  288. // * is the repo name, which can itself be nested
  289. // for example, for GCR this is project-id/repo
  290. // need to use wildcard, see https://github.com/go-chi/chi/issues/243
  291. "/projects/{project_id}/registries/{registry_id}/repositories/*",
  292. auth.DoesUserHaveProjectAccess(
  293. auth.DoesUserHaveRegistryAccess(
  294. requestlog.NewHandler(a.HandleListImages, l),
  295. mw.URLParam,
  296. mw.URLParam,
  297. ),
  298. mw.URLParam,
  299. mw.WriteAccess,
  300. ),
  301. )
  302. // /api/projects/{project_id}/releases routes
  303. r.Method(
  304. "GET",
  305. "/projects/{project_id}/releases",
  306. auth.DoesUserHaveProjectAccess(
  307. auth.DoesUserHaveClusterAccess(
  308. requestlog.NewHandler(a.HandleListReleases, l),
  309. mw.URLParam,
  310. mw.QueryParam,
  311. ),
  312. mw.URLParam,
  313. mw.ReadAccess,
  314. ),
  315. )
  316. r.Method(
  317. "GET",
  318. "/projects/{project_id}/releases/{name}/{revision}/components",
  319. auth.DoesUserHaveProjectAccess(
  320. auth.DoesUserHaveClusterAccess(
  321. requestlog.NewHandler(a.HandleGetReleaseComponents, l),
  322. mw.URLParam,
  323. mw.QueryParam,
  324. ),
  325. mw.URLParam,
  326. mw.ReadAccess,
  327. ),
  328. )
  329. r.Method(
  330. "GET",
  331. "/projects/{project_id}/releases/{name}/{revision}/controllers",
  332. auth.DoesUserHaveProjectAccess(
  333. auth.DoesUserHaveClusterAccess(
  334. requestlog.NewHandler(a.HandleGetReleaseControllers, l),
  335. mw.URLParam,
  336. mw.QueryParam,
  337. ),
  338. mw.URLParam,
  339. mw.ReadAccess,
  340. ),
  341. )
  342. r.Method(
  343. "GET",
  344. "/projects/{project_id}/releases/{name}/history",
  345. auth.DoesUserHaveProjectAccess(
  346. auth.DoesUserHaveClusterAccess(
  347. requestlog.NewHandler(a.HandleListReleaseHistory, l),
  348. mw.URLParam,
  349. mw.QueryParam,
  350. ),
  351. mw.URLParam,
  352. mw.ReadAccess,
  353. ),
  354. )
  355. r.Method(
  356. "POST",
  357. "/projects/{project_id}/releases/{name}/upgrade",
  358. auth.DoesUserHaveProjectAccess(
  359. auth.DoesUserHaveClusterAccess(
  360. requestlog.NewHandler(a.HandleUpgradeRelease, l),
  361. mw.URLParam,
  362. mw.QueryParam,
  363. ),
  364. mw.URLParam,
  365. mw.ReadAccess,
  366. ),
  367. )
  368. r.Method(
  369. "GET",
  370. "/projects/{project_id}/releases/{name}/{revision}",
  371. auth.DoesUserHaveProjectAccess(
  372. auth.DoesUserHaveClusterAccess(
  373. requestlog.NewHandler(a.HandleGetRelease, l),
  374. mw.URLParam,
  375. mw.QueryParam,
  376. ),
  377. mw.URLParam,
  378. mw.ReadAccess,
  379. ),
  380. )
  381. r.Method(
  382. "POST",
  383. "/projects/{project_id}/releases/{name}/rollback",
  384. auth.DoesUserHaveProjectAccess(
  385. auth.DoesUserHaveClusterAccess(
  386. requestlog.NewHandler(a.HandleRollbackRelease, l),
  387. mw.URLParam,
  388. mw.QueryParam,
  389. ),
  390. mw.URLParam,
  391. mw.ReadAccess,
  392. ),
  393. )
  394. // /api/projects/{project_id}/repos routes
  395. // r.Method(
  396. // "GET",
  397. // "/projects/{project_id}/repos",
  398. // auth.DoesUserHaveProjectAccess(
  399. // requestlog.NewHandler(a.HandleListRepos, l),
  400. // mw.URLParam,
  401. // mw.ReadAccess,
  402. // ),
  403. // )
  404. // r.Method(
  405. // "GET",
  406. // "/projects/{project_id}/repos/{kind}/{name}/branches",
  407. // auth.DoesUserHaveProjectAccess(
  408. // requestlog.NewHandler(a.HandleGetBranches, l),
  409. // mw.URLParam,
  410. // mw.ReadAccess,
  411. // ),
  412. // )
  413. // r.Method(
  414. // "GET",
  415. // "/projects/{project_id}/repos/{kind}/{name}/{branch}/contents",
  416. // auth.DoesUserHaveProjectAccess(
  417. // requestlog.NewHandler(a.HandleGetBranchContents, l),
  418. // mw.URLParam,
  419. // mw.ReadAccess,
  420. // ),
  421. // )
  422. // /api/projects/{project_id}/deploy route
  423. r.Method(
  424. "POST",
  425. "/projects/{project_id}/deploy",
  426. auth.DoesUserHaveProjectAccess(
  427. auth.DoesUserHaveClusterAccess(
  428. requestlog.NewHandler(a.HandleDeployTemplate, l),
  429. mw.URLParam,
  430. mw.QueryParam,
  431. ),
  432. mw.URLParam,
  433. mw.ReadAccess,
  434. ),
  435. )
  436. // /api/projects/{project_id}/k8s routes
  437. r.Method(
  438. "GET",
  439. "/projects/{project_id}/k8s/namespaces",
  440. auth.DoesUserHaveProjectAccess(
  441. auth.DoesUserHaveClusterAccess(
  442. requestlog.NewHandler(a.HandleListNamespaces, l),
  443. mw.URLParam,
  444. mw.QueryParam,
  445. ),
  446. mw.URLParam,
  447. mw.ReadAccess,
  448. ),
  449. )
  450. r.Method(
  451. "GET",
  452. "/projects/{project_id}/k8s/{namespace}/pod/{name}/logs",
  453. auth.DoesUserHaveProjectAccess(
  454. auth.DoesUserHaveClusterAccess(
  455. requestlog.NewHandler(a.HandleGetPodLogs, l),
  456. mw.URLParam,
  457. mw.QueryParam,
  458. ),
  459. mw.URLParam,
  460. mw.ReadAccess,
  461. ),
  462. )
  463. r.Method(
  464. "GET",
  465. "/projects/{project_id}/k8s/{kind}/status",
  466. auth.DoesUserHaveProjectAccess(
  467. auth.DoesUserHaveClusterAccess(
  468. requestlog.NewHandler(a.HandleStreamControllerStatus, l),
  469. mw.URLParam,
  470. mw.QueryParam,
  471. ),
  472. mw.URLParam,
  473. mw.ReadAccess,
  474. ),
  475. )
  476. r.Method(
  477. "GET",
  478. "/projects/{project_id}/k8s/pods",
  479. auth.DoesUserHaveProjectAccess(
  480. auth.DoesUserHaveClusterAccess(
  481. requestlog.NewHandler(a.HandleListPods, l),
  482. mw.URLParam,
  483. mw.QueryParam,
  484. ),
  485. mw.URLParam,
  486. mw.ReadAccess,
  487. ),
  488. )
  489. })
  490. fs := http.FileServer(http.Dir(staticFilePath))
  491. r.Get("/*", func(w http.ResponseWriter, r *http.Request) {
  492. if _, err := os.Stat(staticFilePath + r.RequestURI); os.IsNotExist(err) {
  493. http.StripPrefix(r.URL.Path, fs).ServeHTTP(w, r)
  494. } else {
  495. fs.ServeHTTP(w, r)
  496. }
  497. })
  498. return r
  499. }