Compare commits

...

5 Commits

Author SHA1 Message Date
cfd23aa00b
fix: mdoels 2023-11-27 15:59:54 -06:00
d4b937d0f6
chore: stuff 2023-11-27 15:42:33 -06:00
a9d690075e
feat: done! 2023-11-27 15:41:12 -06:00
8c3016e020
fix: pass n to orgreposurl 2023-11-27 13:08:14 -06:00
zachpren0
aea50e7603 gitea repo purs mods 2023-11-27 10:04:42 -08:00
5 changed files with 115 additions and 25 deletions

View File

@ -1,16 +1,19 @@
package: package:
dependencies: dependencies:
- aff - aff
- arrays
- effect - effect
- either - either
- exceptions - exceptions
- fetch - fetch
- foreign-object - foreign-object
- maybe
- mmorph - mmorph
- newtype - newtype
- node-buffer - node-buffer
- prelude - prelude
- simple-json - simple-json
- tailrec
- transformers - transformers
- tuples - tuples
- typelevel-prelude - typelevel-prelude
@ -50,7 +53,7 @@ workspace:
- url - url
url: url:
git: 'https://git.orionkindel.com/thunderstrike/purescript-url-immutable.git' git: 'https://git.orionkindel.com/thunderstrike/purescript-url-immutable.git'
ref: 'dbfa3b6' ref: '555d55a'
dependencies: dependencies:
- arrays - arrays
- effect - effect

View File

@ -3,6 +3,8 @@ module Gitea.Error where
import Prelude import Prelude
import Control.Monad.Error.Class (class MonadThrow, throwError) import Control.Monad.Error.Class (class MonadThrow, throwError)
import Data.Generic.Rep (class Generic)
import Data.Show.Generic (genericShow)
import Effect.Aff.Class (class MonadAff) import Effect.Aff.Class (class MonadAff)
import HTTP.Response as HTTP import HTTP.Response as HTTP
import HTTP.Response as HTTP.Rep import HTTP.Response as HTTP.Rep
@ -13,6 +15,12 @@ data Error
| ENotFound | ENotFound
| EValidation { message :: String, url :: String } | EValidation { message :: String, url :: String }
derive instance Generic Error _
instance Show Error where
show = genericShow
derive instance Eq Error
guardStatusOk :: forall m. MonadThrow Error m => MonadAff m => HTTP.Response -> m Unit guardStatusOk :: forall m. MonadThrow Error m => MonadAff m => HTTP.Response -> m Unit
guardStatusOk rep = do guardStatusOk rep = do
status <- HTTP.Rep.status rep status <- HTTP.Rep.status rep

View File

@ -1,21 +1,18 @@
-- GET /orgs/{org}/repos
-- GET /repos/{owner}/{repo}/topics
-- PUT /repos/{owner}/{repo}/topics/{topic}
-- DELETE /repos/{owner}/{repo}/topics/{topic}
module Gitea.Repo where module Gitea.Repo where
import Prelude hiding ((/)) import Prelude hiding ((/))
import Control.Monad.Reader (ask) import Control.Monad.Reader (ask)
import Control.Monad.Rec.Class (class MonadRec, Step(..), tailRecM)
import Data.Array as Array
import Data.Tuple.Nested ((/\)) import Data.Tuple.Nested ((/\))
import Data.URL (URL, (/)) import Data.URL (URL, (/), (?), (&))
import Effect.Aff.Class (class MonadAff) import Effect.Aff.Class (class MonadAff)
import Effect.Class (liftEffect) import Effect.Class (liftEffect)
import Gitea.Auth as Auth import Gitea.Auth as Auth
import Gitea.Error as Error import Gitea.Error as Error
import Gitea.Trans (GiteaT) import Gitea.Trans (GiteaT)
import Gitea.Types (Repo, RepoName(..), Username(..), RepoUpdate) import Gitea.Types (OrgName(..), Repo, RepoName(..), RepoUpdate, TopicName(..), Username(..))
import HTTP (Method(..), fetch) as HTTP import HTTP (Method(..), fetch) as HTTP
import HTTP.Request (json) as HTTP import HTTP.Request (json) as HTTP
import Node.Buffer (Buffer) import Node.Buffer (Buffer)
@ -25,15 +22,46 @@ import Simple.JSON (class WriteForeign)
import Type.Row.Subset (class Subset) import Type.Row.Subset (class Subset)
reposURL :: forall m. Monad m => Username -> RepoName -> GiteaT m URL reposURL :: forall m. Monad m => Username -> RepoName -> GiteaT m URL
reposURL (Username owner) (RepoName repo) = (\{ baseURI } -> baseURI / "repos" / owner / repo) <$> ask reposURL (Username owner) (RepoName repo) =
let
build { baseURI } = baseURI / "repos" / owner / repo
in
build <$> ask
repoTopicsPageURL :: forall m. Monad m => Username -> RepoName -> { page :: Int, limit :: Int } -> GiteaT m URL
repoTopicsPageURL o repo { page, limit } = (\repos -> repos / "topics" ? "limit" /\ show limit & "page" /\ show page) <$> reposURL o repo
repoTopicsOneURL :: forall m. Monad m => Username -> RepoName -> TopicName -> GiteaT m URL
repoTopicsOneURL o repo (TopicName topic) = (\repos -> repos / "topics" / topic) <$> reposURL o repo
orgReposURL :: forall m. Monad m => OrgName -> { page :: Int, limit :: Int } -> GiteaT m URL
orgReposURL (OrgName org) { page, limit } = (\{ baseURI } -> baseURI / "orgs" / org / "repos" ? "limit" /\ show limit & "page" /\ show page) <$> ask
get :: forall m. MonadAff m => Username -> RepoName -> GiteaT m (Record Repo) get :: forall m. MonadAff m => Username -> RepoName -> GiteaT m (Record Repo)
get u r = do get owner repo = do
{ auth } <- ask { auth } <- ask
url <- reposURL u r url <- reposURL owner repo
rep <- HTTP.fetch $ HTTP.GET /\ url /\ Auth.headers auth rep <- HTTP.fetch $ HTTP.GET /\ url /\ Auth.headers auth
Error.tryGetRepJSON rep Error.tryGetRepJSON rep
listForOrg :: forall m. MonadAff m => MonadRec m => OrgName -> GiteaT m (Array (Record Repo))
listForOrg o =
let
limit = 10
maybeFetchMore repos n newRepos
| Array.length newRepos < limit = Done $ repos <> newRepos
| otherwise = Loop $ (repos <> newRepos) /\ (n + 1)
paginate (repos /\ n) = maybeFetchMore repos n <$> getPage n
getPage n = do
{ auth } <- ask
url <- orgReposURL o { limit, page: n }
rep <- HTTP.fetch $ HTTP.GET /\ url /\ Auth.headers auth
Error.tryGetRepJSON rep
in
tailRecM paginate ([] /\ 1)
update update
:: forall m patch :: forall m patch
. MonadAff m . MonadAff m
@ -43,30 +71,62 @@ update
-> RepoName -> RepoName
-> Record patch -> Record patch
-> GiteaT m (Record Repo) -> GiteaT m (Record Repo)
update u r patch = do update owner repo patch = do
{ auth } <- ask { auth } <- ask
url <- reposURL u r url <- reposURL owner repo
rep <- HTTP.fetch $ HTTP.PATCH /\ url /\ HTTP.json patch /\ Auth.headers auth rep <- HTTP.fetch $ HTTP.PATCH /\ url /\ HTTP.json patch /\ Auth.headers auth
Error.tryGetRepJSON rep Error.tryGetRepJSON rep
delete :: forall m. MonadAff m => Username -> RepoName -> GiteaT m Unit delete :: forall m. MonadAff m => Username -> RepoName -> GiteaT m Unit
delete u r = do delete owner repo = do
{ auth } <- ask { auth } <- ask
url <- reposURL u r url <- reposURL owner repo
rep <- HTTP.fetch $ HTTP.DELETE /\ url /\ Auth.headers auth rep <- HTTP.fetch $ HTTP.DELETE /\ url /\ Auth.headers auth
Error.guardStatusOk rep Error.guardStatusOk rep
updateAvatar :: forall m. MonadAff m => Username -> RepoName -> Buffer -> GiteaT m Unit updateAvatar :: forall m. MonadAff m => Username -> RepoName -> Buffer -> GiteaT m Unit
updateAvatar u r av = do updateAvatar owner repo avatar = do
{ auth } <- ask { auth } <- ask
url <- map (_ / "avatar") $ reposURL u r url <- map (_ / "avatar") $ reposURL owner repo
b64 <- liftEffect $ Buffer.toString Base64 av b64 <- liftEffect $ Buffer.toString Base64 avatar
rep <- HTTP.fetch $ HTTP.POST /\ url /\ HTTP.json { image: b64 } /\ Auth.headers auth rep <- HTTP.fetch $ HTTP.POST /\ url /\ HTTP.json { image: b64 } /\ Auth.headers auth
Error.guardStatusOk rep Error.guardStatusOk rep
removeAvatar :: forall m. MonadAff m => Username -> RepoName -> GiteaT m Unit removeAvatar :: forall m. MonadAff m => Username -> RepoName -> GiteaT m Unit
removeAvatar u r = do removeAvatar owner repo = do
{ auth } <- ask { auth } <- ask
url <- map (_ / "avatar") $ reposURL u r url <- map (_ / "avatar") $ reposURL owner repo
rep <- HTTP.fetch $ HTTP.DELETE /\ url /\ Auth.headers auth
Error.guardStatusOk rep
addTopic :: forall m. MonadAff m => Username -> RepoName -> TopicName -> GiteaT m Unit
addTopic owner repo topic = do
{ auth } <- ask
url <- repoTopicsOneURL owner repo topic
rep <- HTTP.fetch $ HTTP.PUT /\ url /\ Auth.headers auth
Error.guardStatusOk rep
getTopics :: forall m. MonadAff m => MonadRec m => Username -> RepoName -> GiteaT m (Array TopicName)
getTopics owner repo =
let
limit = 10
maybeFetchMore topics n newTopics
| Array.length newTopics < limit = Done $ topics <> newTopics
| otherwise = Loop $ (topics <> newTopics) /\ (n + 1)
paginate (topics /\ n) = maybeFetchMore topics n <$> getPage n
getPage n = do
{ auth } <- ask
url <- repoTopicsPageURL owner repo { limit, page: n }
rep <- HTTP.fetch $ HTTP.GET /\ url /\ Auth.headers auth
Error.tryGetRepJSON rep
in
tailRecM paginate ([] /\ 1)
removeTopic :: forall m. MonadAff m => Username -> RepoName -> TopicName -> GiteaT m Unit
removeTopic owner repo topic = do
{ auth } <- ask
url <- repoTopicsOneURL owner repo topic
rep <- HTTP.fetch $ HTTP.DELETE /\ url /\ Auth.headers auth rep <- HTTP.fetch $ HTTP.DELETE /\ url /\ Auth.headers auth
Error.guardStatusOk rep Error.guardStatusOk rep

View File

@ -6,6 +6,7 @@ import Control.Monad.Error.Class (class MonadError, class MonadThrow, throwError
import Control.Monad.Except (ExceptT, runExceptT) import Control.Monad.Except (ExceptT, runExceptT)
import Control.Monad.Morph (class MFunctor, class MMonad, embed, hoist) import Control.Monad.Morph (class MFunctor, class MMonad, embed, hoist)
import Control.Monad.Reader (class MonadAsk, class MonadReader, ReaderT(..), ask, runReaderT) import Control.Monad.Reader (class MonadAsk, class MonadReader, ReaderT(..), ask, runReaderT)
import Control.Monad.Rec.Class (class MonadRec)
import Control.Monad.Trans.Class (class MonadTrans, lift) import Control.Monad.Trans.Class (class MonadTrans, lift)
import Data.Either (Either) import Data.Either (Either)
import Data.Newtype (class Newtype, unwrap, wrap) import Data.Newtype (class Newtype, unwrap, wrap)
@ -39,6 +40,7 @@ derive newtype instance MonadEffect m => MonadEffect (GiteaT m)
derive newtype instance MonadAff m => MonadAff (GiteaT m) derive newtype instance MonadAff m => MonadAff (GiteaT m)
derive newtype instance Monad m => MonadError Gitea.Error.Error (GiteaT m) derive newtype instance Monad m => MonadError Gitea.Error.Error (GiteaT m)
derive newtype instance Monad m => MonadThrow Gitea.Error.Error (GiteaT m) derive newtype instance Monad m => MonadThrow Gitea.Error.Error (GiteaT m)
derive newtype instance MonadRec m => MonadRec (GiteaT m)
derive newtype instance Monad m => Functor (GiteaT m) derive newtype instance Monad m => Functor (GiteaT m)
derive newtype instance Monad m => Apply (GiteaT m) derive newtype instance Monad m => Apply (GiteaT m)

View File

@ -2,6 +2,7 @@ module Gitea.Types where
import Prelude import Prelude
import Data.Maybe (Maybe)
import Data.Newtype (class Newtype) import Data.Newtype (class Newtype)
import Foreign.Object (Object) import Foreign.Object (Object)
import Simple.JSON (class ReadForeign, class WriteForeign) import Simple.JSON (class ReadForeign, class WriteForeign)
@ -17,6 +18,22 @@ derive newtype instance Show RepoName
derive newtype instance WriteForeign RepoName derive newtype instance WriteForeign RepoName
derive newtype instance ReadForeign RepoName derive newtype instance ReadForeign RepoName
newtype TopicName = TopicName String
derive instance Newtype TopicName _
derive newtype instance Eq TopicName
derive newtype instance Show TopicName
derive newtype instance WriteForeign TopicName
derive newtype instance ReadForeign TopicName
newtype OrgName = OrgName String
derive instance Newtype OrgName _
derive newtype instance Eq OrgName
derive newtype instance Show OrgName
derive newtype instance WriteForeign OrgName
derive newtype instance ReadForeign OrgName
newtype TagName = TagName String newtype TagName = TagName String
derive instance Newtype TagName _ derive instance Newtype TagName _
@ -251,7 +268,7 @@ type RepoGiteaSettingsMutable r =
type RepoSectionGiteaStatsMutable r = type RepoSectionGiteaStatsMutable r =
( "template" :: Boolean ( "template" :: Boolean
, "external_wiki" :: RepoExternalWiki , "external_wiki" :: Maybe RepoExternalWiki
| r | r
) )
@ -264,7 +281,7 @@ type RepoSectionGiteaStatsImmutable r =
, "watchers_count" :: Int , "watchers_count" :: Int
, "fork" :: Boolean , "fork" :: Boolean
, "forks_count" :: Int , "forks_count" :: Int
, "repo_transfer" :: RepoTransfer , "repo_transfer" :: Maybe RepoTransfer
| r | r
) )
@ -295,8 +312,8 @@ type RepoSectionStatsImmutable r =
) )
type RepoSectionTrackerMutable r = type RepoSectionTrackerMutable r =
( "external_tracker" :: RepoExternalTracker ( "external_tracker" :: Maybe RepoExternalTracker
, "internal_tracker" :: RepoInternalTracker , "internal_tracker" :: Maybe RepoInternalTracker
| r | r
) )
@ -354,7 +371,7 @@ type Commit =
, "removed" :: Array String , "removed" :: Array String
, "timestamp" :: String , "timestamp" :: String
, "url" :: String , "url" :: String
, "verification" :: CommitVerification , "verification" :: Maybe CommitVerification
} }
type Branch = type Branch =