From df36dfbb1e79253792917442d9b3636c6d6e6ab7 Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Tue, 27 Jun 2023 16:16:56 -0600 Subject: [PATCH 01/28] Remove share specific endpoints --- .../src/Unison/Server/Share/Definitions.hs | 170 -------- .../src/Unison/Server/Share/FuzzyFind.hs | 364 ------------------ .../src/Unison/Server/Share/RenderDoc.hs | 65 ---- unison-share-api/unison-share-api.cabal | 5 +- 4 files changed, 1 insertion(+), 603 deletions(-) delete mode 100644 unison-share-api/src/Unison/Server/Share/Definitions.hs delete mode 100644 unison-share-api/src/Unison/Server/Share/FuzzyFind.hs delete mode 100644 unison-share-api/src/Unison/Server/Share/RenderDoc.hs diff --git a/unison-share-api/src/Unison/Server/Share/Definitions.hs b/unison-share-api/src/Unison/Server/Share/Definitions.hs deleted file mode 100644 index a56c59f1d1..0000000000 --- a/unison-share-api/src/Unison/Server/Share/Definitions.hs +++ /dev/null @@ -1,170 +0,0 @@ --- | This module contains implementations of Backend methods which are specialized for Share. --- We should likely move them to the Share repository eventually, but for now it's much easier --- to ensure they're resilient to refactors and changes in the Backend API if they live here. --- --- Perhaps we'll move them when the backing implementation switches to postgres. -module Unison.Server.Share.Definitions (definitionForHQName) where - -import Control.Lens hiding ((??)) -import Control.Monad.Except -import Data.Map qualified as Map -import Data.Set qualified as Set -import U.Codebase.Branch qualified as V2Branch -import U.Codebase.Causal qualified as V2Causal -import U.Codebase.HashTags (CausalHash (..)) -import U.Codebase.Sqlite.NameLookups (PathSegments (..), ReversedName (..)) -import U.Codebase.Sqlite.Operations (NamesPerspective (NamesPerspective)) -import U.Codebase.Sqlite.Operations qualified as Ops -import Unison.Codebase (Codebase) -import Unison.Codebase qualified as Codebase -import Unison.Codebase.Path (Path) -import Unison.Codebase.Path qualified as Path -import Unison.Codebase.Runtime qualified as Rt -import Unison.Codebase.SqliteCodebase.Conversions qualified as CV -import Unison.Codebase.SqliteCodebase.Conversions qualified as Cv -import Unison.Debug qualified as Debug -import Unison.HashQualified qualified as HQ -import Unison.LabeledDependency qualified as LD -import Unison.Name (Name) -import Unison.Name qualified as Name -import Unison.NameSegment (NameSegment (..)) -import Unison.Parser.Ann (Ann) -import Unison.Prelude -import Unison.PrettyPrintEnv qualified as PPE -import Unison.PrettyPrintEnvDecl qualified as PPED -import Unison.PrettyPrintEnvDecl.Sqlite qualified as PPESqlite -import Unison.Reference (TermReference) -import Unison.Reference qualified as Reference -import Unison.Referent qualified as Referent -import Unison.Server.Backend hiding (renderDocRefs) -import Unison.Server.Backend qualified as Backend -import Unison.Server.Doc qualified as Doc -import Unison.Server.NameSearch.Sqlite qualified as SqliteNameSearch -import Unison.Server.Share qualified as Share -import Unison.Server.Types -import Unison.Sqlite qualified as Sqlite -import Unison.Symbol (Symbol) -import Unison.Syntax.HashQualified qualified as HQ (toText) -import Unison.Util.Pretty (Width) - --- | Renders a definition for the given name or hash alongside its documentation. -definitionForHQName :: - -- | The path representing the user's current namesRoot. - -- Searches will be limited to definitions within this path, and names will be relative to - -- this path. - Path -> - -- | The root branch to use - CausalHash -> - Maybe Width -> - -- | Whether to suffixify bindings in the rendered syntax - Suffixify -> - -- | Runtime used to evaluate docs. This should be sandboxed if run on the server. - Rt.Runtime Symbol -> - Codebase IO Symbol Ann -> - -- | The name, hash, or both, of the definition to display. - HQ.HashQualified Name -> - Backend IO DefinitionDisplayResults -definitionForHQName perspective rootHash renderWidth suffixifyBindings rt codebase perspectiveQuery = do - result <- liftIO . Codebase.runTransaction codebase $ do - shallowRoot <- resolveCausalHashV2 (Just rootHash) - let rootBranchHash = V2Causal.valueHash shallowRoot - (perspective, perspectiveQuery) <- addNameIfHashOnly codebase perspective perspectiveQuery shallowRoot - (namesPerspective, locatedQuery) <- Share.relocateToNameRoot perspective perspectiveQuery rootBranchHash - pure $ Right (shallowRoot, namesPerspective, locatedQuery) - (shallowRoot, namesPerspective, query) <- either throwError pure result - let namesRoot = Path.fromList . coerce $ Ops.pathToMountedNameLookup namesPerspective - Debug.debugM Debug.Server "definitionForHQName: (namesPerspective, query)" (namesPerspective, query) - -- Bias towards both relative and absolute path to queries, - -- This allows us to still bias towards definitions outside our namesRoot but within the - -- same tree; - -- e.g. if the query is `map` and we're in `base.trunk.List`, - -- we bias towards `map` and `.base.trunk.List.map` which ensures we still prefer names in - -- `trunk` over those in other releases. - -- ppe which returns names fully qualified to the current namesRoot, not to the codebase root. - let biases = maybeToList $ HQ.toName query - let ppedBuilder deps = fmap (PPED.biasTo biases) . liftIO . Codebase.runTransaction codebase $ PPESqlite.ppedForReferences namesPerspective deps - let nameSearch = SqliteNameSearch.nameSearchForPerspective codebase namesPerspective - dr@(DefinitionResults terms types misses) <- liftIO $ Codebase.runTransaction codebase do - definitionsBySuffixes codebase nameSearch DontIncludeCycles [query] - Debug.debugM Debug.Server "definitionForHQName: found definitions" dr - let width = mayDefaultWidth renderWidth - let docResults :: Name -> Backend IO [(HashQualifiedName, UnisonHash, Doc.Doc)] - docResults name = do - Debug.debugM Debug.Server "definitionForHQName: looking up docs for name" name - docRefs <- liftIO $ docsForDefinitionName codebase nameSearch name - Debug.debugM Debug.Server "definitionForHQName: Found these docs" docRefs - renderDocRefs ppedBuilder width codebase rt docRefs - - let drDeps = definitionResultsDependencies dr - termAndTypePPED <- ppedBuilder drDeps - let fqnTermAndTypePPE = PPED.unsuffixifiedPPE termAndTypePPED - typeDefinitions <- - ifor (typesToSyntax suffixifyBindings width termAndTypePPED types) \ref tp -> do - let hqTypeName = PPE.typeNameOrHashOnly fqnTermAndTypePPE ref - docs <- maybe (pure []) docResults (HQ.toName hqTypeName) - mkTypeDefinition codebase termAndTypePPED namesRoot shallowRoot width ref docs tp - termDefinitions <- - ifor (termsToSyntax suffixifyBindings width termAndTypePPED terms) \reference trm -> do - let referent = Referent.Ref reference - let hqTermName = PPE.termNameOrHashOnly fqnTermAndTypePPE referent - docs <- maybe (pure []) docResults (HQ.toName hqTermName) - mkTermDefinition codebase termAndTypePPED namesRoot shallowRoot width reference docs trm - let renderedDisplayTerms = Map.mapKeys Reference.toText termDefinitions - renderedDisplayTypes = Map.mapKeys Reference.toText typeDefinitions - renderedMisses = fmap HQ.toText misses - pure $ - DefinitionDisplayResults - renderedDisplayTerms - renderedDisplayTypes - renderedMisses - --- | A _hopefully_ temporary solution for the following problem: --- --- When rendering definitions by-hash, we don't know which of the project's dependencies we --- may be in, so we don't know which mount to use when rendering it. --- So, first we do a breadth-first recursive search to find some name for that definition, --- then we can use that name to find the mount and render just as we would if provided a name --- up front. -addNameIfHashOnly :: Codebase m v a -> Path -> HQ.HashQualified Name -> V2Branch.CausalBranch Sqlite.Transaction -> Sqlite.Transaction (Path, HQ.HashQualified Name) -addNameIfHashOnly codebase perspective hqQuery rootCausal = case hqQuery of - HQ.HashOnly sh -> do - let rootBranchHash = V2Causal.valueHash rootCausal - let pathSegments = coerce $ Path.toList perspective - startingPerspective@NamesPerspective {pathToMountedNameLookup} <- Ops.namesPerspectiveForRootAndPath rootBranchHash pathSegments - let findTerm = do - termRefs <- lift $ termReferentsByShortHash codebase sh - termRefs - & altMap \ref -> do - MaybeT $ Ops.recursiveTermNameSearch startingPerspective (CV.referent1to2 ref) - let findType = do - typeRefs <- lift $ typeReferencesByShortHash sh - typeRefs - & altMap \ref -> do - MaybeT $ Ops.recursiveTypeNameSearch startingPerspective (Cv.reference1to2 ref) - mayReversedName <- runMaybeT $ findTerm <|> findType - Debug.debugM Debug.Server "addNameIfHashOnly: found reversed name" mayReversedName - pure $ case mayReversedName of - Nothing -> (perspective, hqQuery) - Just fqnReversedName -> - (Path.fromList . coerce $ pathToMountedNameLookup, HQ.NameOnly (Name.fromReverseSegments $ coerce fqnReversedName)) - _ -> pure (perspective, hqQuery) - -renderDocRefs :: - PPEDBuilder -> - Width -> - Codebase IO Symbol Ann -> - Rt.Runtime Symbol -> - [TermReference] -> - Backend IO [(HashQualifiedName, UnisonHash, Doc.Doc)] -renderDocRefs _ppedBuilder _width _codebase _rt [] = pure [] -renderDocRefs ppedBuilder width codebase rt docRefs = do - eDocs <- for docRefs \ref -> (ref,) <$> liftIO (Backend.evalDocRef rt codebase ref) - let docDeps = foldMap (Doc.dependencies . snd) eDocs <> Set.fromList (LD.TermReference <$> docRefs) - docsPPED <- ppedBuilder docDeps - for eDocs \(ref, eDoc) -> do - let name = bestNameForTerm @Symbol (PPED.suffixifiedPPE docsPPED) width (Referent.Ref ref) - let hash = Reference.toText ref - let renderedDoc = Doc.renderDoc docsPPED eDoc - pure (name, hash, renderedDoc) - -type PPEDBuilder = Set LD.LabeledDependency -> Backend IO PPED.PrettyPrintEnvDecl diff --git a/unison-share-api/src/Unison/Server/Share/FuzzyFind.hs b/unison-share-api/src/Unison/Server/Share/FuzzyFind.hs deleted file mode 100644 index 4c5a37bd5e..0000000000 --- a/unison-share-api/src/Unison/Server/Share/FuzzyFind.hs +++ /dev/null @@ -1,364 +0,0 @@ -{-# LANGUAGE DataKinds #-} -{-# LANGUAGE DeriveAnyClass #-} -{-# LANGUAGE DeriveGeneric #-} -{-# LANGUAGE KindSignatures #-} -{-# LANGUAGE OverloadedStrings #-} -{-# LANGUAGE StandaloneDeriving #-} -{-# LANGUAGE TypeOperators #-} -{-# LANGUAGE ViewPatterns #-} -{-# OPTIONS_GHC -Wno-orphans #-} - -module Unison.Server.Share.FuzzyFind where - -import Control.Monad.Except -import Data.Aeson -import Data.Char qualified as Char -import Data.List qualified as List -import Data.OpenApi (ToSchema) -import Data.Text qualified as Text -import Servant - ( QueryParam, - (:>), - ) -import Servant.Docs - ( DocQueryParam (..), - ParamKind (Normal), - ToParam (..), - ToSample (..), - noSamples, - ) -import Servant.OpenApi () -import U.Codebase.Causal qualified as V2Causal -import U.Codebase.HashTags (BranchHash, CausalHash) -import U.Codebase.Sqlite.NameLookups (PathSegments (..)) -import U.Codebase.Sqlite.NameLookups qualified as NameLookups -import U.Codebase.Sqlite.NamedRef qualified as S -import U.Codebase.Sqlite.Operations qualified as SqliteOps -import Unison.Codebase (Codebase) -import Unison.Codebase qualified as Codebase -import Unison.Codebase.Editor.DisplayObject -import Unison.Codebase.Path qualified as Path -import Unison.Codebase.ShortCausalHash qualified as SCH -import Unison.Codebase.SqliteCodebase.Conversions qualified as Cv -import Unison.NameSegment -import Unison.Parser.Ann (Ann) -import Unison.Prelude -import Unison.PrettyPrintEnvDecl qualified as PPED -import Unison.PrettyPrintEnvDecl.Sqlite qualified as PPED -import Unison.Server.Backend (termEntryLabeledDependencies, typeEntryLabeledDependencies) -import Unison.Server.Backend qualified as Backend -import Unison.Server.Syntax (SyntaxText) -import Unison.Server.Types - ( APIGet, - ExactName (..), - HashQualifiedName, - NamedTerm, - NamedType, - UnisonName, - mayDefaultWidth, - ) -import Unison.Symbol (Symbol) -import Unison.Util.Pretty (Width) - -type FuzzyFindAPI = - "find" - :> QueryParam "rootBranch" SCH.ShortCausalHash - :> QueryParam "relativeTo" Path.Path - :> QueryParam "limit" Int - :> QueryParam "renderWidth" Width - :> QueryParam "query" String - :> APIGet [(Alignment, FoundResult)] - -instance ToSample Alignment where - toSamples _ = noSamples - -instance ToParam (QueryParam "limit" Int) where - toParam _ = - DocQueryParam - "limit" - ["1", "10", "20"] - "The maximum number of results to return. Defaults to 10." - Normal - -instance ToParam (QueryParam "query" String) where - toParam _ = - DocQueryParam - "query" - ["foo", "ff", "td nr"] - "Space-separated subsequences to find in the name of a type or term." - Normal - -data FoundTerm = FoundTerm - { bestFoundTermName :: HashQualifiedName, - namedTerm :: NamedTerm - } - deriving (Generic, Show) - -data FoundType = FoundType - { bestFoundTypeName :: HashQualifiedName, - typeDef :: DisplayObject SyntaxText SyntaxText, - namedType :: NamedType - } - deriving (Generic, Show) - -instance ToJSON FoundType where - toJSON (FoundType {bestFoundTypeName, typeDef, namedType}) = - object - [ "bestFoundTypeName" .= bestFoundTypeName, - "typeDef" .= typeDef, - "namedType" .= namedType - ] - -deriving instance ToSchema FoundType - -instance ToJSON FoundTerm where - toJSON (FoundTerm {bestFoundTermName, namedTerm}) = - object - [ "bestFoundTermName" .= bestFoundTermName, - "namedTerm" .= namedTerm - ] - -deriving instance ToSchema FoundTerm - -data FoundResult - = FoundTermResult FoundTerm - | FoundTypeResult FoundType - deriving (Generic, Show) - -instance ToJSON FoundResult where - toJSON = \case - FoundTermResult ft -> object ["tag" .= String "FoundTermResult", "contents" .= ft] - FoundTypeResult ft -> object ["tag" .= String "FoundTypeResult", "contents" .= ft] - -deriving instance ToSchema FoundResult - -instance ToSample FoundResult where - toSamples _ = noSamples - -serveFuzzyFind :: - -- | Whether the root is a scratch root - Bool -> - -- | Whether to search in dependencies - Bool -> - Codebase IO Symbol Ann -> - CausalHash -> - Path.Path -> - Maybe Int -> - Maybe Width -> - Text -> - Backend.Backend IO [(Alignment, FoundResult)] -serveFuzzyFind inScratch searchDependencies codebase rootCausal perspective mayLimit typeWidth query = do - (includeDependencies, bh, namesPerspective, dbTermMatches, dbTypeMatches) <- liftIO . Codebase.runTransaction codebase $ do - shallowRoot <- Backend.resolveCausalHashV2 (Just rootCausal) - let bh = V2Causal.valueHash shallowRoot - namesPerspective@SqliteOps.NamesPerspective {pathToMountedNameLookup = PathSegments pathToPerspective} <- SqliteOps.namesPerspectiveForRootAndPath bh (coerce $ Path.toList perspective) - -- If were browsing at a scratch root we need to include one level of dependencies even if - -- the 'include-dependencies' flag is not set - -- since the projects are all "dependencies" of the scratch root as far as name-lookups - -- are concerned. - let isScratchRootSearch = inScratch && null pathToPerspective - -- Include dependencies if they were explicitly requested OR if we're running a search - -- from a scratch root - let includeDependencies = isScratchRootSearch || searchDependencies - (terms, types) <- SqliteOps.fuzzySearchDefinitions includeDependencies namesPerspective limit preparedQuery - pure (includeDependencies, bh, namesPerspective, terms, types) - let prepareMatch :: S.NamedRef Backend.FoundRef -> (PathSegments, Alignment, UnisonName, [Backend.FoundRef]) - prepareMatch name@(S.NamedRef {S.reversedSegments}) = - let renderedName = NameLookups.reversedNameToNamespaceText reversedSegments - segments = computeMatchSegments preparedQuery name - alignment = Alignment {score = scoreMatch name, result = MatchResult {segments}} - in (NameLookups.reversedNameToPathSegments reversedSegments, alignment, renderedName, [S.ref name]) - let preparedTerms :: [(PathSegments, Alignment, UnisonName, [Backend.FoundRef])] - preparedTerms = - dbTermMatches - <&> \match -> - match - & fmap (\(ref, ct) -> Backend.FoundTermRef $ Cv.referent2to1UsingCT (fromMaybe (error "serveFuzzyFind: CT required but not found") ct) ref) - & prepareMatch - let preparedTypes :: [(PathSegments, Alignment, UnisonName, [Backend.FoundRef])] - preparedTypes = prepareMatch . fmap (Backend.FoundTypeRef . Cv.reference2to1) <$> dbTypeMatches - let alignments :: - ( [ ( PathSegments, - Alignment, - UnisonName, - [Backend.FoundRef] - ) - ] - ) - alignments = - (preparedTerms <> preparedTypes) - & List.sortOn (\(_, Alignment {score}, _, _) -> score) - lift (join <$> traverse (loadEntry includeDependencies bh namesPerspective) alignments) - where - preparedQuery = prepareQuery (Text.unpack query) - limit = fromMaybe 10 mayLimit - loadEntry :: Bool -> BranchHash -> SqliteOps.NamesPerspective -> (PathSegments, Alignment, Text, [Backend.FoundRef]) -> IO [(Alignment, FoundResult)] - loadEntry includeDependencies bh searchPerspective (pathToMatch, a, n, refs) = do - namesPerspective <- - -- If we're including dependencies we need to ensure each match's type signature is - -- rendered using a ppe with that dependency's names. - -- So we re-compute the perspective for each match. - -- - -- If not we can use the same perspective for every match. - if includeDependencies - then Codebase.runTransaction codebase $ SqliteOps.namesPerspectiveForRootAndPath bh (coerce (Path.toList perspective) <> pathToMatch) - else pure searchPerspective - let relativeToBranch = Nothing - entries <- for refs $ - \case - Backend.FoundTermRef r -> do - Left . (r,) <$> Backend.termListEntry codebase relativeToBranch (ExactName (NameSegment n) (Cv.referent1to2 r)) - Backend.FoundTypeRef r -> - Codebase.runTransaction codebase do - Right . (r,) <$> Backend.typeListEntry codebase relativeToBranch (ExactName (NameSegment n) r) - let allLabeledDependencies = foldMap (either (termEntryLabeledDependencies . snd) (typeEntryLabeledDependencies . snd)) entries - pped <- liftIO . Codebase.runTransaction codebase $ PPED.ppedForReferences namesPerspective allLabeledDependencies - let ppe = PPED.suffixifiedPPE pped - Codebase.runTransaction codebase do - for entries \case - Left (r, termEntry) -> - pure - ( a, - FoundTermResult - . FoundTerm - (Backend.bestNameForTerm @Symbol ppe (mayDefaultWidth typeWidth) r) - $ Backend.termEntryToNamedTerm ppe typeWidth termEntry - ) - Right (r, typeEntry) -> do - let namedType = Backend.typeEntryToNamedType typeEntry - let typeName = Backend.bestNameForType @Symbol ppe (mayDefaultWidth typeWidth) r - typeHeader <- Backend.typeDeclHeader codebase ppe r - let ft = FoundType typeName typeHeader namedType - pure (a, FoundTypeResult ft) - --- Scores a matched name by the number of segments. --- Lower is better. -scoreMatch :: S.NamedRef r -> Int -scoreMatch S.NamedRef {S.reversedSegments = NameLookups.ReversedName segments} = length segments - -data Alignment = Alignment - { score :: Int, - result :: MatchResult - } - deriving stock (Generic) - deriving anyclass (ToSchema) - -data MatchResult = MatchResult - { segments :: [MatchSegment] - } - deriving stock (Generic) - deriving anyclass (ToSchema) - -data MatchSegment - = Gap Text - | Match Text - deriving stock (Show, Generic) - deriving anyclass (ToSchema) - -instance ToJSON Alignment where - toJSON (Alignment {score, result}) = - object ["score" .= score, "result" .= result] - -instance ToJSON MatchResult where - toJSON (MatchResult {segments}) = object ["segments" .= toJSON segments] - -instance ToJSON MatchSegment where - toJSON = \case - Gap s -> object ["tag" .= String "Gap", "contents" .= s] - Match s -> object ["tag" .= String "Match", "contents" .= s] - --- After finding a search results with fuzzy find we do some post processing to --- refine the result: --- * Sort: --- we sort both on the FZF score and the number of segments in the FQN --- preferring shorter FQNs over longer. This helps with things like forks --- of base. --- * Dedupe: --- we dedupe on the found refs to avoid having several rows of a --- definition with different names in the result set. --- --- >>> import qualified Data.List.NonEmpty as NonEmpty --- >>> computeMatchSegments ["foo", "baz"] (S.NamedRef (NameLookups.ReversedName ("baz" NonEmpty.:| ["bar", "foo"])) ()) --- [Match "foo",Gap ".bar.",Match "baz"] --- --- >>> computeMatchSegments ["Li", "Ma"] (S.NamedRef (NameLookups.ReversedName ("foldMap" NonEmpty.:| ["List", "data"])) ()) --- [Gap "data.",Match "Li",Gap "st.fold",Match "Ma",Gap "p"] -computeMatchSegments :: - [Text] -> - (S.NamedRef r) -> - [MatchSegment] -computeMatchSegments query (S.NamedRef {reversedSegments}) = - let nameText = NameLookups.reversedNameToNamespaceText reversedSegments - -- This will be a list of _lower-cased_ match segments, but we need to reclaim the - -- casing from the actual name. - matchSegmentShape = List.unfoldr splitIntoSegments (filter (not . Text.null) . map Text.toLower $ query, Text.toLower nameText) - in List.unfoldr reCasifySegments (matchSegmentShape, nameText) - where - -- The actual matching is case-insensitive but we want to preserve the casing of the - -- actual name, so we use the size of match segments to segment the actual name which has - -- the correct case. - reCasifySegments :: ([MatchSegment], Text) -> Maybe (MatchSegment, ([MatchSegment], Text)) - reCasifySegments = \case - ([], _) -> Nothing - (Gap gap : restShape, name) -> - let (actualGap, restName) = Text.splitAt (Text.length gap) name - in Just (Gap actualGap, (restShape, restName)) - (Match match : restShape, name) -> - let (actualMatch, restName) = Text.splitAt (Text.length match) name - in Just (Match actualMatch, (restShape, restName)) - -- Using the query, split the match into chunks of 'match' or 'gap' - splitIntoSegments :: ([Text], Text) -> Maybe (MatchSegment, ([Text], Text)) - splitIntoSegments = \case - (_, "") -> Nothing - ([], rest) -> Just (Gap rest, ([], "")) - (q : qs, name) -> - Text.breakOn q name - & \case - ("", rest) -> - case Text.stripPrefix q rest of - Nothing -> Nothing - Just remainder -> - Just (Match q, (qs, remainder)) - (gap, rest) -> - Just (Gap gap, (q : qs, rest)) - --- | Splits a query into segments, where each segment must appear in order in any matching --- names. --- --- >>> prepareQuery "foo bar baz" --- ["foo","bar","baz"] --- --- Split camel-case style words into segments. --- >>> prepareQuery "fMap" --- ["f","Map"] --- --- Collapse multiple spaces --- >>> prepareQuery "foo barBaz boom" --- ["foo","bar","Baz","boom"] --- --- Split namespaces into segments with a required dot in between. --- >>> prepareQuery "List.map" --- ["List",".","map"] --- --- Shouldn't get multiple splits for capitalized letters --- >>> prepareQuery "List.Map" --- ["List",".","Map"] -prepareQuery :: String -> [Text] -prepareQuery query = do - word <- words query - xs <- - word - & List.foldl' - ( \acc next -> case next of - c - | Char.isUpper c -> [c] : acc - | Char.isSpace c -> "" : acc - | c == '.' -> "" : "." : acc - | otherwise -> case acc of - [] -> [[c]] - (last : rest) -> (last ++ [c]) : rest - ) - [] - & reverse - & filter (not . null) - pure $ Text.pack xs diff --git a/unison-share-api/src/Unison/Server/Share/RenderDoc.hs b/unison-share-api/src/Unison/Server/Share/RenderDoc.hs deleted file mode 100644 index ba8506917c..0000000000 --- a/unison-share-api/src/Unison/Server/Share/RenderDoc.hs +++ /dev/null @@ -1,65 +0,0 @@ -{-# LANGUAGE DataKinds #-} -{-# LANGUAGE DeriveAnyClass #-} -{-# LANGUAGE DeriveGeneric #-} -{-# LANGUAGE KindSignatures #-} -{-# LANGUAGE OverloadedStrings #-} -{-# LANGUAGE StandaloneDeriving #-} -{-# LANGUAGE TypeOperators #-} -{-# OPTIONS_GHC -Wno-orphans #-} - --- | Helper for rendering docs within a given namespace -module Unison.Server.Share.RenderDoc where - -import Control.Monad.Except -import Servant.OpenApi () -import U.Codebase.Causal qualified as V2Causal -import U.Codebase.HashTags (CausalHash) -import Unison.Codebase (Codebase) -import Unison.Codebase qualified as Codebase -import Unison.Codebase.Path qualified as Path -import Unison.Codebase.Runtime qualified as Rt -import Unison.Codebase.ShortCausalHash (ShortCausalHash) -import Unison.NameSegment (NameSegment) -import Unison.Parser.Ann (Ann) -import Unison.Prelude -import Unison.Server.Backend -import Unison.Server.Backend qualified as Backend -import Unison.Server.Doc (Doc) -import Unison.Server.Types - ( mayDefaultWidth, - ) -import Unison.Symbol (Symbol) -import Unison.Util.Pretty (Width) - -renderDoc :: - Set NameSegment -> - Rt.Runtime Symbol -> - Codebase IO Symbol Ann -> - Path.Path -> - Maybe (Either ShortCausalHash CausalHash) -> - Maybe Width -> - Backend IO (Maybe Doc) -renderDoc docNames runtime codebase namespacePath mayRoot mayWidth = - let width = mayDefaultWidth mayWidth - in do - (rootCausal, shallowBranch) <- - Backend.hoistBackend (Codebase.runTransaction codebase) do - rootCausalHash <- - case mayRoot of - Nothing -> Backend.resolveRootBranchHashV2 Nothing - Just (Left sch) -> Backend.resolveRootBranchHashV2 (Just sch) - Just (Right ch) -> lift $ Backend.resolveCausalHashV2 (Just ch) - -- lift (Backend.resolveCausalHashV2 rootCausalHash) - namespaceCausal <- lift $ Codebase.getShallowCausalAtPath namespacePath (Just rootCausalHash) - shallowBranch <- lift $ V2Causal.value namespaceCausal - pure (rootCausalHash, shallowBranch) - (_localNamesOnly, ppe) <- Backend.scopedNamesForBranchHash codebase (Just rootCausal) namespacePath - renderedDoc <- - Backend.findDocInBranchAndRender - docNames - width - runtime - codebase - ppe - shallowBranch - pure renderedDoc diff --git a/unison-share-api/unison-share-api.cabal b/unison-share-api/unison-share-api.cabal index 5c59a6f940..a9adfb8e7f 100644 --- a/unison-share-api/unison-share-api.cabal +++ b/unison-share-api/unison-share-api.cabal @@ -1,6 +1,6 @@ cabal-version: 1.12 --- This file has been generated from package.yaml by hpack version 0.35.1. +-- This file has been generated from package.yaml by hpack version 0.35.2. -- -- see: https://github.com/sol/hpack @@ -40,9 +40,6 @@ library Unison.Server.SearchResult Unison.Server.SearchResult' Unison.Server.Share - Unison.Server.Share.Definitions - Unison.Server.Share.FuzzyFind - Unison.Server.Share.RenderDoc Unison.Server.Syntax Unison.Server.Types Unison.Sync.API From b678b330e1a087081382c28ef66354df81dac415 Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Wed, 28 Jun 2023 11:54:11 -0600 Subject: [PATCH 02/28] Relax type in backend module --- unison-share-api/src/Unison/Server/Backend.hs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/unison-share-api/src/Unison/Server/Backend.hs b/unison-share-api/src/Unison/Server/Backend.hs index aa7e19e88d..fecc87de33 100644 --- a/unison-share-api/src/Unison/Server/Backend.hs +++ b/unison-share-api/src/Unison/Server/Backend.hs @@ -880,6 +880,7 @@ mungeSyntaxText :: mungeSyntaxText = fmap Syntax.convertElement mkTypeDefinition :: + MonadIO m => Codebase IO Symbol Ann -> PPED.PrettyPrintEnvDecl -> Path.Path -> @@ -890,7 +891,7 @@ mkTypeDefinition :: DisplayObject (AnnotatedText (UST.Element Reference)) (AnnotatedText (UST.Element Reference)) -> - Backend IO TypeDefinition + m TypeDefinition mkTypeDefinition codebase pped namesRoot rootCausal width r docs tp = do let bn = bestNameForType @Symbol (PPED.suffixifiedPPE pped) width r tag <- From bb756da02f77669e792e37a392a32638ef143d71 Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Thu, 29 Jun 2023 11:46:15 -0600 Subject: [PATCH 03/28] Move some helpers to enlil --- unison-share-api/src/Unison/Server/Share.hs | 39 --------------------- unison-share-api/unison-share-api.cabal | 1 - 2 files changed, 40 deletions(-) delete mode 100644 unison-share-api/src/Unison/Server/Share.hs diff --git a/unison-share-api/src/Unison/Server/Share.hs b/unison-share-api/src/Unison/Server/Share.hs deleted file mode 100644 index 84748d0dbd..0000000000 --- a/unison-share-api/src/Unison/Server/Share.hs +++ /dev/null @@ -1,39 +0,0 @@ --- | Helpers which are specific to the remote share server. -module Unison.Server.Share (relocateToNameRoot) where - -import Control.Lens hiding ((??)) -import Data.List.NonEmpty qualified as NonEmpty -import U.Codebase.HashTags (BranchHash) -import U.Codebase.Sqlite.NameLookups (PathSegments (..)) -import U.Codebase.Sqlite.Operations (NamesPerspective (..)) -import U.Codebase.Sqlite.Operations qualified as Ops -import Unison.Codebase.Path (Path) -import Unison.Codebase.Path qualified as Path -import Unison.Debug qualified as Debug -import Unison.HashQualified qualified as HQ -import Unison.Name (Name) -import Unison.Name qualified as Name -import Unison.NameSegment (NameSegment (..)) -import Unison.Prelude -import Unison.Sqlite qualified as Sqlite - --- | Given an arbitrary query and perspective, find the name root the query belongs in, --- then return that root and the query relocated to that root. --- --- A name root is either a project root or a dependency root. --- E.g. @.myproject.some.namespace -> .myproject@ or @.myproject.lib.base.List -> .myproject.lib.base@ -relocateToNameRoot :: Path -> HQ.HashQualified Name -> BranchHash -> Sqlite.Transaction (NamesPerspective, HQ.HashQualified Name) -relocateToNameRoot perspective query rootBh = do - -- The namespace containing the name path - let nameLocation = case HQ.toName query of - Just name -> - name - & Name.segments - & NonEmpty.init - & Path.fromList - Nothing -> Path.empty - let fullPath = perspective <> nameLocation - Debug.debugM Debug.Server "relocateToNameRoot fullPath" fullPath - namesPerspective@NamesPerspective {relativePerspective} <- Ops.namesPerspectiveForRootAndPath rootBh (PathSegments . coerce . Path.toList $ fullPath) - let reprefixName name = Name.fromReverseSegments $ (NonEmpty.head $ Name.reverseSegments name) NonEmpty.:| (reverse $ coerce relativePerspective) - pure (namesPerspective, reprefixName <$> query) diff --git a/unison-share-api/unison-share-api.cabal b/unison-share-api/unison-share-api.cabal index a9adfb8e7f..7053d647f7 100644 --- a/unison-share-api/unison-share-api.cabal +++ b/unison-share-api/unison-share-api.cabal @@ -39,7 +39,6 @@ library Unison.Server.QueryResult Unison.Server.SearchResult Unison.Server.SearchResult' - Unison.Server.Share Unison.Server.Syntax Unison.Server.Types Unison.Sync.API From c68e14bddfb46106e97345e04cc1e623ddd63466 Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Wed, 5 Jul 2023 10:48:59 -0600 Subject: [PATCH 04/28] WIP --- unison-share-api/src/Unison/Server/Backend.hs | 1 + 1 file changed, 1 insertion(+) diff --git a/unison-share-api/src/Unison/Server/Backend.hs b/unison-share-api/src/Unison/Server/Backend.hs index fecc87de33..a00d7ff4db 100644 --- a/unison-share-api/src/Unison/Server/Backend.hs +++ b/unison-share-api/src/Unison/Server/Backend.hs @@ -85,6 +85,7 @@ module Unison.Server.Backend evalDocRef, mkTermDefinition, mkTypeDefinition, + displayTerm, ) where From 5102a10737f0b0ab9c9ba61adc3921da30885e0f Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Fri, 7 Jul 2023 16:40:02 -0600 Subject: [PATCH 05/28] Move NamespaceDetails module to Enlil --- .../Unison/Server/Share/NamespaceDetails.hs | 38 ------------------- unison-share-api/unison-share-api.cabal | 2 - 2 files changed, 40 deletions(-) delete mode 100644 unison-share-api/src/Unison/Server/Share/NamespaceDetails.hs diff --git a/unison-share-api/src/Unison/Server/Share/NamespaceDetails.hs b/unison-share-api/src/Unison/Server/Share/NamespaceDetails.hs deleted file mode 100644 index df4caaab00..0000000000 --- a/unison-share-api/src/Unison/Server/Share/NamespaceDetails.hs +++ /dev/null @@ -1,38 +0,0 @@ -module Unison.Server.Share.NamespaceDetails (namespaceDetails) where - -import Control.Monad.Except -import Data.Set qualified as Set -import Servant.OpenApi () -import U.Codebase.HashTags (CausalHash) -import Unison.Codebase (Codebase) -import Unison.Codebase qualified as Codebase -import Unison.Codebase.Path qualified as Path -import Unison.Codebase.Runtime qualified as Rt -import Unison.Parser.Ann (Ann) -import Unison.Server.Backend -import Unison.Server.Backend qualified as Backend -import Unison.Server.Share.RenderDoc qualified as RenderDoc -import Unison.Server.Types - ( NamespaceDetails (..), - v2CausalBranchToUnisonHash, - ) -import Unison.Symbol (Symbol) -import Unison.Util.Pretty (Width) - -namespaceDetails :: - Rt.Runtime Symbol -> - Codebase IO Symbol Ann -> - Path.Path -> - CausalHash -> - Maybe Width -> - Backend IO NamespaceDetails -namespaceDetails runtime codebase namespacePath rootCausalHash mayWidth = do - causalHashAtPath <- liftIO $ Codebase.runTransaction codebase do - causalBranch <- Backend.resolveCausalHashV2 (Just rootCausalHash) - namespaceCausal <- Codebase.getShallowCausalAtPath namespacePath (Just causalBranch) - let causalHashAtPath = v2CausalBranchToUnisonHash namespaceCausal - pure causalHashAtPath - mayReadme <- RenderDoc.findAndRenderDoc readmeNames runtime codebase namespacePath rootCausalHash mayWidth - pure $ NamespaceDetails namespacePath causalHashAtPath mayReadme - where - readmeNames = Set.fromList ["README", "Readme", "ReadMe", "readme"] diff --git a/unison-share-api/unison-share-api.cabal b/unison-share-api/unison-share-api.cabal index 5fe680f9c5..5fc6fef765 100644 --- a/unison-share-api/unison-share-api.cabal +++ b/unison-share-api/unison-share-api.cabal @@ -39,8 +39,6 @@ library Unison.Server.QueryResult Unison.Server.SearchResult Unison.Server.SearchResult' - Unison.Server.Share.NamespaceDetails - Unison.Server.Share.RenderDoc Unison.Server.Syntax Unison.Server.Types Unison.Sync.API From 56ae975033c014d82d7cfd551b331c5a40a625ca Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Tue, 11 Jul 2023 14:10:09 -0600 Subject: [PATCH 06/28] Hoist tree diffs --- parser-typechecker/src/U/Codebase/Branch/Diff.hs | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/parser-typechecker/src/U/Codebase/Branch/Diff.hs b/parser-typechecker/src/U/Codebase/Branch/Diff.hs index c66a3f23d0..197d03b3b2 100644 --- a/parser-typechecker/src/U/Codebase/Branch/Diff.hs +++ b/parser-typechecker/src/U/Codebase/Branch/Diff.hs @@ -1,5 +1,6 @@ module U.Codebase.Branch.Diff ( TreeDiff (..), + hoistTreeDiff, NameChanges (..), DefinitionDiffs (..), Diff (..), @@ -12,6 +13,7 @@ module U.Codebase.Branch.Diff where import Control.Comonad.Cofree +import Control.Comonad.Cofree qualified as Cofree import Control.Lens (ifoldMap) import Control.Lens qualified as Lens import Data.Functor.Compose (Compose (..)) @@ -76,6 +78,10 @@ instance (Applicative m) => Semigroup (TreeDiff m) where instance (Applicative m) => Monoid (TreeDiff m) where mempty = TreeDiff (mempty :< Compose mempty) +hoistTreeDiff :: Functor m => (forall x. m x -> n x) -> TreeDiff m -> TreeDiff n +hoistTreeDiff f (TreeDiff cfr) = + TreeDiff $ Cofree.hoistCofree (\(Compose m) -> Compose (fmap f m)) cfr + -- | A summary of a 'TreeDiff', containing all names added and removed. -- Note that there isn't a clear notion of a name "changing" since conflicts might muddy the notion -- by having multiple copies of both the from and to names, so we just talk about adds and From 74d69d816137dd53a014edd3c8eaa47631f7e02a Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Tue, 18 Jul 2023 12:35:57 -0600 Subject: [PATCH 07/28] Clean empty children when computing diff --- .../src/U/Codebase/Branch/Diff.hs | 33 ++++++++++++------- .../Codebase/SqliteCodebase/Operations.hs | 2 +- 2 files changed, 22 insertions(+), 13 deletions(-) diff --git a/parser-typechecker/src/U/Codebase/Branch/Diff.hs b/parser-typechecker/src/U/Codebase/Branch/Diff.hs index 197d03b3b2..028c5ef515 100644 --- a/parser-typechecker/src/U/Codebase/Branch/Diff.hs +++ b/parser-typechecker/src/U/Codebase/Branch/Diff.hs @@ -22,6 +22,7 @@ import Data.Semialign qualified as Align import Data.Set qualified as Set import Data.These import U.Codebase.Branch +import U.Codebase.Branch qualified as V2Branch import U.Codebase.Branch.Type qualified as Branch import U.Codebase.Causal qualified as Causal import U.Codebase.Reference (Reference) @@ -31,6 +32,7 @@ import Unison.Name (Name) import Unison.Name qualified as Name import Unison.NameSegment (NameSegment) import Unison.Prelude +import Unison.Sqlite qualified as Sqlite import Unison.Util.Monoid (foldMapM, ifoldMapM) import Unison.Util.Relation (Relation) import Unison.Util.Relation qualified as Relation @@ -120,24 +122,26 @@ instance Semigroup NameBasedDiff where NameBasedDiff (terms0 <> terms1) (types0 <> types1) -- | Diff two Branches, returning a tree containing all of the changes -diffBranches :: forall m. (Monad m) => Branch m -> Branch m -> TreeDiff m -diffBranches from to = +diffBranches :: Branch Sqlite.Transaction -> Branch Sqlite.Transaction -> Sqlite.Transaction (TreeDiff Sqlite.Transaction) +diffBranches from to = do + fromChildren <- V2Branch.nonEmptyChildren from + toChildren <- V2Branch.nonEmptyChildren to let termDiffs = diffMap (Branch.terms from) (Branch.terms to) - typeDiffs = diffMap (Branch.types from) (Branch.types to) - defDiff = DefinitionDiffs {termDiffs, typeDiffs} - childDiff :: (Map NameSegment (m (Cofree (Compose (Map NameSegment) m) DefinitionDiffs))) - childDiff = do - Align.align (children from) (children to) + let typeDiffs = diffMap (Branch.types from) (Branch.types to) + let defDiff = DefinitionDiffs {termDiffs, typeDiffs} + let childDiff :: Map NameSegment (Sqlite.Transaction (Cofree (Compose (Map NameSegment) Sqlite.Transaction) DefinitionDiffs)) + childDiff = + Align.align fromChildren toChildren & mapMaybe \case This ca -> Just do -- TODO: For the names index we really don't need to know which exact -- names were removed, we just need to delete from the index using a -- prefix query, this would be faster than crawling to get all the deletes. removedChildBranch <- Causal.value ca - pure . unTreeDiff $ diffBranches removedChildBranch Branch.empty + unTreeDiff <$> diffBranches removedChildBranch Branch.empty That ca -> Just do newChildBranch <- Causal.value ca - pure . unTreeDiff $ diffBranches Branch.empty newChildBranch + unTreeDiff <$> diffBranches Branch.empty newChildBranch These fromC toC | Causal.valueHash fromC == Causal.valueHash toC -> -- This child didn't change. @@ -145,12 +149,13 @@ diffBranches from to = | otherwise -> Just $ do fromChildBranch <- Causal.value fromC toChildBranch <- Causal.value toC - case diffBranches fromChildBranch toChildBranch of + diffBranches fromChildBranch toChildBranch >>= \case TreeDiff (defDiffs :< Compose mchildren) -> do pure $ (defDiffs :< Compose mchildren) - in TreeDiff (defDiff :< Compose childDiff) + pure $ + TreeDiff (defDiff :< Compose childDiff) where - diffMap :: forall ref. (Ord ref) => Map NameSegment (Map ref (m MdValues)) -> Map NameSegment (Map ref (m MdValues)) -> Map NameSegment (Diff ref) + diffMap :: forall ref. (Ord ref) => Map NameSegment (Map ref (Sqlite.Transaction MdValues)) -> Map NameSegment (Map ref (Sqlite.Transaction MdValues)) -> Map NameSegment (Diff ref) diffMap l r = Align.align l r & fmap \case @@ -218,6 +223,10 @@ streamNameChanges namePrefix (TreeDiff (DefinitionDiffs {termDiffs, typeDiffs} : in (listifyNames name $ adds diff, listifyNames name $ removals diff) let nameChanges = NameChanges {termNameAdds, termNameRemovals, typeNameAdds, typeNameRemovals} acc <- f namePrefix nameChanges + -- acc <- + -- if nameChanges == mempty + -- then pure mempty + -- else f namePrefix nameChanges childAcc <- children & ifoldMapM diff --git a/parser-typechecker/src/Unison/Codebase/SqliteCodebase/Operations.hs b/parser-typechecker/src/Unison/Codebase/SqliteCodebase/Operations.hs index 62f76f4f96..043b44304d 100644 --- a/parser-typechecker/src/Unison/Codebase/SqliteCodebase/Operations.hs +++ b/parser-typechecker/src/Unison/Codebase/SqliteCodebase/Operations.hs @@ -648,7 +648,7 @@ ensureNameLookupForBranchHash getDeclType mayFromBranchHash toBranchHash = do toBranch <- Ops.expectBranchByBranchHash toBranchHash depMounts <- Projects.inferDependencyMounts toBranch <&> fmap (first (coerce @_ @PathSegments . Path.toList)) let depMountPaths = (Path.fromList . coerce) . fst <$> depMounts - let treeDiff = ignoreDepMounts depMountPaths $ BranchDiff.diffBranches fromBranch toBranch + treeDiff <- ignoreDepMounts depMountPaths <$> BranchDiff.diffBranches fromBranch toBranch let namePrefix = Nothing Ops.buildNameLookupForBranchHash mayExistingLookupBH From caebdbaccc65ac11c58f308a2d76b36004f00555 Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Mon, 14 Aug 2023 16:22:49 -0700 Subject: [PATCH 08/28] Apply new diffing types to merged in code --- unison-cli/src/Unison/Codebase/Editor/HandleInput.hs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/unison-cli/src/Unison/Codebase/Editor/HandleInput.hs b/unison-cli/src/Unison/Codebase/Editor/HandleInput.hs index eb9b026ce3..f06d974b5d 100644 --- a/unison-cli/src/Unison/Codebase/Editor/HandleInput.hs +++ b/unison-cli/src/Unison/Codebase/Editor/HandleInput.hs @@ -1390,7 +1390,7 @@ loop e = do Cli.runTransaction do fromBranch <- Codebase.expectCausalBranchByCausalHash fromCH >>= V2Causal.value toBranch <- Codebase.expectCausalBranchByCausalHash toCH >>= V2Causal.value - let treeDiff = V2Branch.Diff.diffBranches fromBranch toBranch + treeDiff <- V2Branch.Diff.diffBranches fromBranch toBranch nameChanges <- V2Branch.Diff.allNameChanges Nothing treeDiff pure (DisplayDebugNameDiff nameChanges) Cli.respond output @@ -1979,7 +1979,7 @@ handleDiffNamespaceToPatch description input = do branch1 <- ExceptT (Cli.resolveAbsBranchIdV2 absBranchId1) branch2 <- ExceptT (Cli.resolveAbsBranchIdV2 absBranchId2) lift do - branchDiff <- V2Branch.Diff.nameBasedDiff (V2Branch.Diff.diffBranches branch1 branch2) + branchDiff <- V2Branch.Diff.diffBranches branch1 branch2 >>= V2Branch.Diff.nameBasedDiff termEdits <- (branchDiff ^. #terms) & Relation.domain From 0798b17b25a6f0afbcb3790a7058df726e64ec04 Mon Sep 17 00:00:00 2001 From: Jared Forsyth Date: Tue, 29 Aug 2023 09:14:06 -0500 Subject: [PATCH 09/28] [iops] updated base I guess --- unison-src/builtin-tests/base.output.md | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/unison-src/builtin-tests/base.output.md b/unison-src/builtin-tests/base.output.md index 98378ddfc9..85d33115ef 100644 --- a/unison-src/builtin-tests/base.output.md +++ b/unison-src/builtin-tests/base.output.md @@ -5,9 +5,9 @@ Thus, make sure the contents of this file define the contents of the cache (e.g. don't pull `latest`.) ```ucm -.> pull @unison/base/releases/2.2.0 .base +.> pull @unison/base/releases/2.5.0 .base - Downloaded 12209 entities. + Downloaded 12426 entities. ✅ @@ -15,11 +15,11 @@ Thus, make sure the contents of this file define the contents of the cache .> compile.native.fetch - Downloaded 1255 entities. + Downloaded 1549 entities. ✅ Successfully updated .unison.internal from - @unison/internal/releases/0.0.1. + @unison/internal/releases/0.0.3. ``` From 8014451080f14162b7f4ba5b3427485a717b67ca Mon Sep 17 00:00:00 2001 From: Jared Forsyth Date: Tue, 29 Aug 2023 09:14:06 -0500 Subject: [PATCH 10/28] [iops] ok pulling is working --- scheme-libs/racket/unison/io-handles.rkt | 37 +++++++++--------------- scheme-libs/racket/unison/io.rkt | 18 ++++++++++++ scheme-libs/racket/unison/primops.ss | 5 ++++ unison-src/builtin-tests/base.output.md | 12 ++++---- unison-src/builtin-tests/io-tests.u | 10 +++++++ unison-src/builtin-tests/jit-tests.sh | 3 ++ 6 files changed, 55 insertions(+), 30 deletions(-) diff --git a/scheme-libs/racket/unison/io-handles.rkt b/scheme-libs/racket/unison/io-handles.rkt index a093ccf4f2..ab779f79b5 100644 --- a/scheme-libs/racket/unison/io-handles.rkt +++ b/scheme-libs/racket/unison/io-handles.rkt @@ -1,33 +1,15 @@ #lang racket/base -(require racket/exn - racket/string - racket/file +(require racket/string rnrs/io/ports-6 (only-in rnrs standard-error-port standard-input-port standard-output-port vector-map) (only-in racket empty? with-output-to-string system/exit-code system false?) - compatibility/mlist (only-in unison/boot data-case define-unison) unison/data unison/chunked-seq - unison/core - unison/tcp - unison/pem - unison/core - unison/data - unison/data-info - unison/math - unison/chunked-seq - unison/chunked-bytes - unison/bytes-nat - unison/pattern - unison/crypto - unison/data - unison/io - unison/tls - unison/tcp - unison/gzip - unison/zlib - unison/concurrent + unison/data + unison/data-info + unison/chunked-seq + unison/data ) (provide @@ -46,6 +28,8 @@ getEnv.impl.v1 getChar.impl.v1 isFileOpen.impl.v3 + isSeekable.impl.v3 + handlePosition.impl.v3 process.call getCurrentDirectory.impl.v3 ready.impl.v1 @@ -78,6 +62,13 @@ (unison-either-right (string->chunked-string (path->string (current-directory))))) +(define-unison (isSeekable.impl.v3 handle) + (unison-either-right + (if (port-has-set-port-position!? handle) unison-boolean-false unison-boolean-true))) + +(define-unison (handlePosition.impl.v3 handle) + (unison-either-right (port-position handle))) + (define-unison (seekHandle.impl.v3 handle mode amount) (data-case mode (0 () diff --git a/scheme-libs/racket/unison/io.rkt b/scheme-libs/racket/unison/io.rkt index ce4a9babf3..2b0a8ed7bd 100644 --- a/scheme-libs/racket/unison/io.rkt +++ b/scheme-libs/racket/unison/io.rkt @@ -35,6 +35,9 @@ renameFile.impl.v3 createDirectory.impl.v3 removeDirectory.impl.v3 + setCurrentDirectory.impl.v3 + renameDirectory.impl.v3 + isDirectory.impl.v3 createTempDirectory.impl.v3))) (define (getFileSize.impl.v3 path) @@ -62,6 +65,10 @@ (define (getTempDirectory.impl.v3) (right (string->chunked-string (path->string (find-system-path 'temp-dir))))) +(define-unison (setCurrentDirectory.impl.v3 path) + (current-directory (chunked-string->string path)) + (unison-either-right none)) + (define-unison (createTempDirectory.impl.v3 prefix) (unison-either-right (string->chunked-string @@ -78,6 +85,17 @@ (delete-directory (chunked-string->string file)) (unison-either-right none)) +(define-unison (isDirectory.impl.v3 path) + (unison-either-right + (if (directory-exists? (chunked-string->string path)) + unison-boolean-true + unison-boolean-false))) + +(define-unison (renameDirectory.impl.v3 old new) + (rename-file-or-directory (chunked-string->string old) + (chunked-string->string new)) + (unison-either-right none)) + (define-unison (renameFile.impl.v3 old new) (rename-file-or-directory (chunked-string->string old) (chunked-string->string new)) diff --git a/scheme-libs/racket/unison/primops.ss b/scheme-libs/racket/unison/primops.ss index 62c1cabf3c..b9f3a66c20 100644 --- a/scheme-libs/racket/unison/primops.ss +++ b/scheme-libs/racket/unison/primops.ss @@ -88,6 +88,11 @@ builtin-IO.renameFile.impl.v3 builtin-IO.createTempDirectory.impl.v3 builtin-IO.createDirectory.impl.v3 + builtin-IO.setCurrentDirectory.impl.v3 + builtin-IO.renameDirectory.impl.v3 + builtin-IO.isDirectory.impl.v3 + builtin-IO.isSeekable.impl.v3 + builtin-IO.handlePosition.impl.v3 unison-FOp-IO.getFileSize.impl.v3 unison-FOp-IO.getFileTimestamp.impl.v3 unison-FOp-IO.fileExists.impl.v3 diff --git a/unison-src/builtin-tests/base.output.md b/unison-src/builtin-tests/base.output.md index 85d33115ef..0d4f73ad90 100644 --- a/unison-src/builtin-tests/base.output.md +++ b/unison-src/builtin-tests/base.output.md @@ -7,19 +7,17 @@ Thus, make sure the contents of this file define the contents of the cache ```ucm .> pull @unison/base/releases/2.5.0 .base - Downloaded 12426 entities. + Merging... - ✅ + 😶 - Successfully pulled into .base, which was empty. + .base was already up-to-date with @unison/base/releases/2.5.0. .> compile.native.fetch - Downloaded 1549 entities. - - ✅ + 😶 - Successfully updated .unison.internal from + .unison.internal was already up-to-date with @unison/internal/releases/0.0.3. ``` diff --git a/unison-src/builtin-tests/io-tests.u b/unison-src/builtin-tests/io-tests.u index 7e7315e4ad..fef91c1153 100644 --- a/unison-src/builtin-tests/io-tests.u +++ b/unison-src/builtin-tests/io-tests.u @@ -19,6 +19,7 @@ io.tests = Tests.main do !io.test_renameFile !io.test_isFileOpen !io.test_ready + !io.test_now rm_if_exists fp = if FilePath.exists fp @@ -32,6 +33,15 @@ testFile = do rm_if_exists fp fp +io.test_now = do + match !now with + Instant a b -> + if a Int.> +10 then + Tests.pass "!now is working" + else + Tests.fail "Now is too small" "yeah" + + io.test_createTempDirectory = do tmp = (createTempDirectory (FilePath "prefix-")) match tmp with diff --git a/unison-src/builtin-tests/jit-tests.sh b/unison-src/builtin-tests/jit-tests.sh index e471f4fa69..b49196f431 100755 --- a/unison-src/builtin-tests/jit-tests.sh +++ b/unison-src/builtin-tests/jit-tests.sh @@ -8,6 +8,9 @@ base_codebase=${XDG_CACHE_HOME:-"$HOME/.cache"}/unisonlanguage/base.unison if [ ! -d $base_codebase ]; then echo !!!! Creating a codebase in $base_codebase $ucm transcript -S $base_codebase unison-src/builtin-tests/base.md +else + echo !!!! Updating the codebase in $base_codebase + $ucm transcript.fork -c $base_codebase -S $base_codebase unison-src/builtin-tests/base.md fi dir=${XDG_DATA_HOME:-"$HOME/.local/share"}/unisonlanguage/scheme-libs From 6aa0c7e29a4a17b92a542727c245f9502157b09a Mon Sep 17 00:00:00 2001 From: Jared Forsyth Date: Tue, 29 Aug 2023 09:14:06 -0500 Subject: [PATCH 11/28] [iops] not working lol --- scheme-libs/racket/unison/io.rkt | 8 ++++++++ scheme-libs/racket/unison/primops.ss | 2 ++ unison-src/builtin-tests/io-tests.u | 20 ++++++++++++++++++-- unison-src/builtin-tests/jit-tests.output.md | 20 +++++++++++++++----- 4 files changed, 43 insertions(+), 7 deletions(-) diff --git a/scheme-libs/racket/unison/io.rkt b/scheme-libs/racket/unison/io.rkt index 2b0a8ed7bd..d8b41758ca 100644 --- a/scheme-libs/racket/unison/io.rkt +++ b/scheme-libs/racket/unison/io.rkt @@ -38,6 +38,8 @@ setCurrentDirectory.impl.v3 renameDirectory.impl.v3 isDirectory.impl.v3 + systemTime.impl.v3 + systemTimeMicroseconds.impl.v3 createTempDirectory.impl.v3))) (define (getFileSize.impl.v3 path) @@ -101,6 +103,12 @@ (chunked-string->string new)) (unison-either-right none)) +(define-unison (systemTime.impl.v3 unit) + (unison-either-right (current-seconds))) + +(define-unison (systemTimeMicroseconds.impl.v3 unit) + (unison-either-right (inexact->exact (* 1000 (current-inexact-milliseconds))))) + (define (threadCPUTime.v1) (right (current-process-milliseconds (current-thread)))) (define (processCPUTime.v1) diff --git a/scheme-libs/racket/unison/primops.ss b/scheme-libs/racket/unison/primops.ss index b9f3a66c20..be1cee3e83 100644 --- a/scheme-libs/racket/unison/primops.ss +++ b/scheme-libs/racket/unison/primops.ss @@ -93,6 +93,8 @@ builtin-IO.isDirectory.impl.v3 builtin-IO.isSeekable.impl.v3 builtin-IO.handlePosition.impl.v3 + builtin-IO.systemTime.impl.v3 + builtin-IO.systemTimeMicroseconds.impl.v3 unison-FOp-IO.getFileSize.impl.v3 unison-FOp-IO.getFileTimestamp.impl.v3 unison-FOp-IO.fileExists.impl.v3 diff --git a/unison-src/builtin-tests/io-tests.u b/unison-src/builtin-tests/io-tests.u index fef91c1153..335d404a81 100644 --- a/unison-src/builtin-tests/io-tests.u +++ b/unison-src/builtin-tests/io-tests.u @@ -33,13 +33,29 @@ testFile = do rm_if_exists fp fp +io.test_deprecated_systemTimeMicroseconds = do + match !systemTimeMicroseconds with + micro -> + if micro Int.> +10 then + Tests.pass "!systemTimeMicroseconds" + else + Tests.fail "!systemTimeMicroseconds" "systemTime is too small" + +io.test_deprecated_systemTime = do + match !systemTime with + EpochTime seconds -> + if seconds Int.> +10 then + Tests.pass "!systemTime" + else + Tests.fail "!systemTime" "systemTime is too small" + io.test_now = do match !now with Instant a b -> if a Int.> +10 then - Tests.pass "!now is working" + Tests.pass "!now" else - Tests.fail "Now is too small" "yeah" + Tests.fail "!now" "now is too small" io.test_createTempDirectory = do diff --git a/unison-src/builtin-tests/jit-tests.output.md b/unison-src/builtin-tests/jit-tests.output.md index f43952ca6f..759e879209 100644 --- a/unison-src/builtin-tests/jit-tests.output.md +++ b/unison-src/builtin-tests/jit-tests.output.md @@ -6,10 +6,20 @@ then reference those tests (which should be of type `'{IO,Exception,Tests} ()`, to `Tests.check` and `Tests.checkEqual`). ```ucm -.> run.native tests - +.> alias.term ##IO.randomBytes IO.randomBytes.> load unison-src/builtin-tests/io-tests.u.> add ``` -```ucm -.> run.native tests.jit.only -``` + +🛑 + +The transcript failed due to an error in the stanza above. The error is: + + + The 1st argument to `(Int.>)` + + has type: Nat + but I expected: Int + + 47 | if seconds Int.> +10 then + + From 2339fae497c1d5a412cbcba549d3d5d7bedad471 Mon Sep 17 00:00:00 2001 From: Arya Irani Date: Tue, 29 Aug 2023 10:31:06 -0400 Subject: [PATCH 12/28] try to set a custom workflow name on release --- .github/workflows/release.yaml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.github/workflows/release.yaml b/.github/workflows/release.yaml index 6ca80242dd..7233da23b7 100644 --- a/.github/workflows/release.yaml +++ b/.github/workflows/release.yaml @@ -1,5 +1,7 @@ name: "release" +run-name: "release ${{inputs.version}}" + defaults: run: shell: bash From e64aa0ec5eaf97f6b8fef278c2f830b4faf4fb55 Mon Sep 17 00:00:00 2001 From: Jared Forsyth Date: Wed, 30 Aug 2023 06:58:47 -0500 Subject: [PATCH 13/28] [iops] ok new tests passing --- unison-src/builtin-tests/io-tests.u | 53 ++++++++++++++++++++++++++++- 1 file changed, 52 insertions(+), 1 deletion(-) diff --git a/unison-src/builtin-tests/io-tests.u b/unison-src/builtin-tests/io-tests.u index 335d404a81..a2f0baccfe 100644 --- a/unison-src/builtin-tests/io-tests.u +++ b/unison-src/builtin-tests/io-tests.u @@ -20,6 +20,10 @@ io.tests = Tests.main do !io.test_isFileOpen !io.test_ready !io.test_now + !io.test_isSeekable + !io.test_handlePosition + !io.test_renameDirectory + !io.test_setCurrentDirectory rm_if_exists fp = if FilePath.exists fp @@ -28,11 +32,33 @@ rm_if_exists fp = else () +rmdir_if_exists fp = + if FilePath.exists fp + then + removeDirectory fp + else + () + + testFile = do fp = FilePath ((FilePath.toText !getTempDirectory) ++ "/unison-test") rm_if_exists fp fp +io.test_isSeekable = do + fp = !testFile + fhandle = open fp Write + checkEqual "isSeekable file" true (isSeekable fhandle) + +io.test_handlePosition = do + fp = !testFile + writeFile fp "123456" + fhandle = open fp Read + checkEqual "handlePosition initial" 0 (position fhandle) + _ = getBytes fhandle 2 + checkEqual "handlePosition" 2 (position fhandle) + + io.test_deprecated_systemTimeMicroseconds = do match !systemTimeMicroseconds with micro -> @@ -44,7 +70,7 @@ io.test_deprecated_systemTimeMicroseconds = do io.test_deprecated_systemTime = do match !systemTime with EpochTime seconds -> - if seconds Int.> +10 then + if seconds > 10 then Tests.pass "!systemTime" else Tests.fail "!systemTime" "systemTime is too small" @@ -87,6 +113,23 @@ io.test_ready = do ready1 = ready fhandle checkEqual "handle with text ready is ready" ready1 true +join fp text = + FilePath (FilePath.toText fp ++ "/" ++ text) + +io.test_renameDirectory = do + td = join !getTempDirectory "unison-dir" + rd = join !getTempDirectory "unison-dir-rename" + rmdir_if_exists td + rmdir_if_exists rd + createDirectory td + contents = "a file contents" + _ = writeFile (join td "hello") contents + renameDirectory td rd + got = (getText (open (join rd "hello") Read)) + rmdir_if_exists td + rmdir_if_exists rd + checkEqual "renameFile" contents got + io.test_renameFile = do fp = FilePath ((FilePath.toText !getTempDirectory) ++ "/unison-test") rmp = FilePath ((FilePath.toText !getTempDirectory) ++ "/unison-test-renamed") @@ -133,6 +176,14 @@ test_getFileTimestamp = do test_getFileTimestamp_err = do expectError' "File timestamp of missing file" ["does not exist", "error getting"] '(FilePath.getTimestamp !testFile) +io.test_setCurrentDirectory = do + prev = !getCurrentDirectory + setCurrentDirectory (FilePath "/") + new = !getCurrentDirectory + setCurrentDirectory prev + match new with + FilePath text -> checkEqual "Current directory set & get" "/" text + io.test_getCurrentDirectory = do match !getCurrentDirectory with FilePath text -> if Text.startsWith "/" text then From 994b3e512b4cb3fec76738edb3b3a8542a41cdc1 Mon Sep 17 00:00:00 2001 From: Jared Forsyth Date: Wed, 30 Aug 2023 07:00:45 -0500 Subject: [PATCH 14/28] [iops] tests passing interpreter --- unison-src/builtin-tests/io-tests.u | 20 ++++++++------------ 1 file changed, 8 insertions(+), 12 deletions(-) diff --git a/unison-src/builtin-tests/io-tests.u b/unison-src/builtin-tests/io-tests.u index a2f0baccfe..7c06fb7646 100644 --- a/unison-src/builtin-tests/io-tests.u +++ b/unison-src/builtin-tests/io-tests.u @@ -28,14 +28,10 @@ io.tests = Tests.main do rm_if_exists fp = if FilePath.exists fp then - removeFile fp - else - () - -rmdir_if_exists fp = - if FilePath.exists fp - then - removeDirectory fp + if isDirectory fp then + removeDirectory fp + else + removeFile fp else () @@ -119,15 +115,15 @@ join fp text = io.test_renameDirectory = do td = join !getTempDirectory "unison-dir" rd = join !getTempDirectory "unison-dir-rename" - rmdir_if_exists td - rmdir_if_exists rd + rm_if_exists td + rm_if_exists rd createDirectory td contents = "a file contents" _ = writeFile (join td "hello") contents renameDirectory td rd got = (getText (open (join rd "hello") Read)) - rmdir_if_exists td - rmdir_if_exists rd + rm_if_exists td + rm_if_exists rd checkEqual "renameFile" contents got io.test_renameFile = do From 09a30e49f869861d3f559251f570224f3bc0a513 Mon Sep 17 00:00:00 2001 From: Jared Forsyth Date: Wed, 30 Aug 2023 07:02:25 -0500 Subject: [PATCH 15/28] [iops] nice --- scheme-libs/racket/unison/io-handles.rkt | 2 +- scheme-libs/racket/unison/io.rkt | 2 +- unison-src/builtin-tests/jit-tests.output.md | 20 +++++--------------- 3 files changed, 7 insertions(+), 17 deletions(-) diff --git a/scheme-libs/racket/unison/io-handles.rkt b/scheme-libs/racket/unison/io-handles.rkt index ab779f79b5..3160e96e45 100644 --- a/scheme-libs/racket/unison/io-handles.rkt +++ b/scheme-libs/racket/unison/io-handles.rkt @@ -64,7 +64,7 @@ (define-unison (isSeekable.impl.v3 handle) (unison-either-right - (if (port-has-set-port-position!? handle) unison-boolean-false unison-boolean-true))) + (if (port-has-set-port-position!? handle) unison-boolean-true unison-boolean-false))) (define-unison (handlePosition.impl.v3 handle) (unison-either-right (port-position handle))) diff --git a/scheme-libs/racket/unison/io.rkt b/scheme-libs/racket/unison/io.rkt index d8b41758ca..7f4edd3175 100644 --- a/scheme-libs/racket/unison/io.rkt +++ b/scheme-libs/racket/unison/io.rkt @@ -84,7 +84,7 @@ (unison-either-right none)) (define-unison (removeDirectory.impl.v3 file) - (delete-directory (chunked-string->string file)) + (delete-directory/files (chunked-string->string file)) (unison-either-right none)) (define-unison (isDirectory.impl.v3 path) diff --git a/unison-src/builtin-tests/jit-tests.output.md b/unison-src/builtin-tests/jit-tests.output.md index 759e879209..f43952ca6f 100644 --- a/unison-src/builtin-tests/jit-tests.output.md +++ b/unison-src/builtin-tests/jit-tests.output.md @@ -6,20 +6,10 @@ then reference those tests (which should be of type `'{IO,Exception,Tests} ()`, to `Tests.check` and `Tests.checkEqual`). ```ucm -.> alias.term ##IO.randomBytes IO.randomBytes.> load unison-src/builtin-tests/io-tests.u.> add -``` - - -🛑 - -The transcript failed due to an error in the stanza above. The error is: +.> run.native tests +``` +```ucm +.> run.native tests.jit.only - The 1st argument to `(Int.>)` - - has type: Nat - but I expected: Int - - 47 | if seconds Int.> +10 then - - +``` From 101b48e1ed8dc9e8956f5b5e2a3ae2fb139ad722 Mon Sep 17 00:00:00 2001 From: Travis Staton Date: Tue, 8 Aug 2023 13:18:11 -0400 Subject: [PATCH 16/28] add haskell-language-server-wrapper symlink --- flake.nix | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/flake.nix b/flake.nix index 231e488f03..73bfcd2d61 100644 --- a/flake.nix +++ b/flake.nix @@ -119,6 +119,13 @@ ormolu = { version = ormolu-ver; }; haskell-language-server = { version = "latest"; + modules = [ + { + packages.haskell-language-server.components.exes.haskell-language-server.postInstall = '' + ln -sr "$out/bin/haskell-language-server" "$out/bin/haskell-language-server-wrapper" + ''; + } + ]; # specify flags via project file rather than a module override # https://github.com/input-output-hk/haskell.nix/issues/1509 cabalProject = '' From c3e9a424720e33a6901fdd84c4d66c3d279a47c4 Mon Sep 17 00:00:00 2001 From: Mitchell Rosen Date: Thu, 31 Aug 2023 11:07:32 -0400 Subject: [PATCH 17/28] add back some of old ShortHash comment --- codebase2/core/Unison/ShortHash.hs | 20 ++++++++++++++------ 1 file changed, 14 insertions(+), 6 deletions(-) diff --git a/codebase2/core/Unison/ShortHash.hs b/codebase2/core/Unison/ShortHash.hs index 36bbc8d319..98532cb928 100644 --- a/codebase2/core/Unison/ShortHash.hs +++ b/codebase2/core/Unison/ShortHash.hs @@ -18,13 +18,21 @@ where import Data.Text qualified as Text import Unison.Prelude +-- A ShortHash is used to query the Codebase for anonymous definitions. The prefix should look like base32hex, but is +-- not decoded here because the prefix doesn't correspond to anything useful - we'll just compare strings against the +-- codebase later. +-- -- ##Text.++ --- ^^^^^^^-- builtin - --- #abc123.a#0 --- ^ ^ ^-cid --- | \-cycle --- \-- prefix +-- ^^^^^^^ +-- | +-- builtin +-- +-- #abc123.1#2 +-- ^^^^^^ ^ ^ +-- | | | +-- | | cid +-- | cycle +-- prefix data ShortHash = Builtin Text | ShortHash {prefix :: Text, cycle :: Maybe Word64, cid :: Maybe Word64} From fee42779f3ab78260af1a98ca16787364f81cf11 Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Thu, 31 Aug 2023 15:11:18 -0700 Subject: [PATCH 18/28] Apply perf optimization: don't call callback unless there are changes --- parser-typechecker/src/U/Codebase/Branch/Diff.hs | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/parser-typechecker/src/U/Codebase/Branch/Diff.hs b/parser-typechecker/src/U/Codebase/Branch/Diff.hs index 028c5ef515..e816daf034 100644 --- a/parser-typechecker/src/U/Codebase/Branch/Diff.hs +++ b/parser-typechecker/src/U/Codebase/Branch/Diff.hs @@ -222,11 +222,10 @@ streamNameChanges namePrefix (TreeDiff (DefinitionDiffs {termDiffs, typeDiffs} : let name = appendName ns in (listifyNames name $ adds diff, listifyNames name $ removals diff) let nameChanges = NameChanges {termNameAdds, termNameRemovals, typeNameAdds, typeNameRemovals} - acc <- f namePrefix nameChanges - -- acc <- - -- if nameChanges == mempty - -- then pure mempty - -- else f namePrefix nameChanges + acc <- + if nameChanges == mempty + then pure mempty + else f namePrefix nameChanges childAcc <- children & ifoldMapM From 7e22abd06611a741b8a1aee6ce1dcf7b0e5c3a30 Mon Sep 17 00:00:00 2001 From: sixfourtwelve Date: Tue, 5 Sep 2023 16:48:50 +0100 Subject: [PATCH 19/28] add extra help to delete --- unison-cli/src/Unison/CommandLine/InputPatterns.hs | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/unison-cli/src/Unison/CommandLine/InputPatterns.hs b/unison-cli/src/Unison/CommandLine/InputPatterns.hs index a267834c7c..e0a5b8d525 100644 --- a/unison-cli/src/Unison/CommandLine/InputPatterns.hs +++ b/unison-cli/src/Unison/CommandLine/InputPatterns.hs @@ -686,13 +686,19 @@ deleteGen :: Maybe String -> String -> ([Path.HQSplit'] -> DeleteTarget) -> Inpu deleteGen suffix target mkTarget = let cmd = maybe "delete" ("delete." <>) suffix info = - P.sep + P.wrapColumn2 [ + (P.sep " " [ backtick (P.sep " " [P.string cmd, "foo"]), "removes the", P.string target, "name `foo` from the namespace." - ] + ], ""), + (P.sep + " " + [ backtick (P.sep " " [P.string cmd, "foo bar"]), + P.string target, + "name `foo` and `bar` from the namespace" ], "")] warn = P.sep " " From fbd02b0e325ae7f41c0884bad34cf8413d7c9c04 Mon Sep 17 00:00:00 2001 From: sixfourtwelve Date: Tue, 5 Sep 2023 16:57:42 +0100 Subject: [PATCH 20/28] slight typo --- unison-cli/src/Unison/CommandLine/InputPatterns.hs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/unison-cli/src/Unison/CommandLine/InputPatterns.hs b/unison-cli/src/Unison/CommandLine/InputPatterns.hs index e0a5b8d525..d46e80d909 100644 --- a/unison-cli/src/Unison/CommandLine/InputPatterns.hs +++ b/unison-cli/src/Unison/CommandLine/InputPatterns.hs @@ -698,7 +698,7 @@ deleteGen suffix target mkTarget = " " [ backtick (P.sep " " [P.string cmd, "foo bar"]), P.string target, - "name `foo` and `bar` from the namespace" ], "")] + "remotes the name `foo` and `bar` from the namespace" ], "")] warn = P.sep " " From eb24c5a7c0a50c32c36e47e240b7a7d7bcb3bfd0 Mon Sep 17 00:00:00 2001 From: sixfourtwelve Date: Tue, 5 Sep 2023 16:59:35 +0100 Subject: [PATCH 21/28] slight typo --- unison-cli/src/Unison/CommandLine/InputPatterns.hs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/unison-cli/src/Unison/CommandLine/InputPatterns.hs b/unison-cli/src/Unison/CommandLine/InputPatterns.hs index d46e80d909..54df811493 100644 --- a/unison-cli/src/Unison/CommandLine/InputPatterns.hs +++ b/unison-cli/src/Unison/CommandLine/InputPatterns.hs @@ -697,8 +697,9 @@ deleteGen suffix target mkTarget = (P.sep " " [ backtick (P.sep " " [P.string cmd, "foo bar"]), + "removes the", P.string target, - "remotes the name `foo` and `bar` from the namespace" ], "")] + "name `foo` and `bar` from the namespace" ], "")] warn = P.sep " " From 934e10ecc06e50933af44ada9342c70e1d30f595 Mon Sep 17 00:00:00 2001 From: Arya Irani <538571+aryairani@users.noreply.github.com> Date: Tue, 5 Sep 2023 15:58:10 -0400 Subject: [PATCH 22/28] Update unison-cli/src/Unison/CommandLine/InputPatterns.hs --- unison-cli/src/Unison/CommandLine/InputPatterns.hs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/unison-cli/src/Unison/CommandLine/InputPatterns.hs b/unison-cli/src/Unison/CommandLine/InputPatterns.hs index 54df811493..a3f6d18170 100644 --- a/unison-cli/src/Unison/CommandLine/InputPatterns.hs +++ b/unison-cli/src/Unison/CommandLine/InputPatterns.hs @@ -699,7 +699,7 @@ deleteGen suffix target mkTarget = [ backtick (P.sep " " [P.string cmd, "foo bar"]), "removes the", P.string target, - "name `foo` and `bar` from the namespace" ], "")] + "name `foo` and `bar` from the namespace." ], "")] warn = P.sep " " From b52170832a572eb27df12fa8e8b6aa1bd02cd5b2 Mon Sep 17 00:00:00 2001 From: sixfourtwelve Date: Tue, 5 Sep 2023 23:00:03 +0100 Subject: [PATCH 23/28] add logic to flip hint --- .../U/Codebase/Sqlite/Queries.hs | 7 ++++++ unison-cli/src/Unison/CommandLine/Welcome.hs | 24 +++++++++++-------- unison-cli/unison/Main.hs | 6 ++--- 3 files changed, 24 insertions(+), 13 deletions(-) diff --git a/codebase2/codebase-sqlite/U/Codebase/Sqlite/Queries.hs b/codebase2/codebase-sqlite/U/Codebase/Sqlite/Queries.hs index 03791bd811..5326da548f 100644 --- a/codebase2/codebase-sqlite/U/Codebase/Sqlite/Queries.hs +++ b/codebase2/codebase-sqlite/U/Codebase/Sqlite/Queries.hs @@ -107,6 +107,7 @@ module U.Codebase.Sqlite.Queries -- * projects projectExists, + doProjectsExist, projectExistsByName, loadProject, loadProjectByName, @@ -3224,6 +3225,12 @@ projectExists projectId = ) |] +-- | Check if any projects exist +doProjectsExist :: Transaction Bool +doProjectsExist = + queryOneCol + [sql| SELECT EXISTS (SELECT 1 FROM project) |] + -- | Does a project exist by this name? projectExistsByName :: ProjectName -> Transaction Bool projectExistsByName name = diff --git a/unison-cli/src/Unison/CommandLine/Welcome.hs b/unison-cli/src/Unison/CommandLine/Welcome.hs index f5b9a4b104..5418f0524d 100644 --- a/unison-cli/src/Unison/CommandLine/Welcome.hs +++ b/unison-cli/src/Unison/CommandLine/Welcome.hs @@ -14,7 +14,8 @@ import Prelude hiding (readFile, writeFile) data Welcome = Welcome { onboarding :: Onboarding, -- Onboarding States - unisonVersion :: Text + unisonVersion :: Text, + welcomeHint :: Bool } -- Previously Created is different from Previously Onboarded because a user can @@ -34,12 +35,12 @@ data Onboarding | PreviouslyOnboarded deriving (Show, Eq) -welcome :: CodebaseInitStatus -> Text -> Welcome -welcome initStatus unisonVersion = - Welcome (Init initStatus) unisonVersion +welcome :: CodebaseInitStatus -> Text -> Bool -> Welcome +welcome initStatus unisonVersion welcomeHint = + Welcome (Init initStatus) unisonVersion welcomeHint run :: Welcome -> [Either Event Input] -run Welcome {onboarding = onboarding, unisonVersion = version} = do +run Welcome {onboarding = onboarding, unisonVersion = version, welcomeHint = welcomeHint} = do go onboarding [] where go :: Onboarding -> [Either Event Input] -> [Either Event Input] @@ -58,8 +59,8 @@ run Welcome {onboarding = onboarding, unisonVersion = version} = do where authorMsg = toInput authorSuggestion -- These are our two terminal Welcome conditions, at the end we reverse the order of the desired input commands otherwise they come out backwards - Finished -> reverse (toInput getStarted : acc) - PreviouslyOnboarded -> reverse (toInput getStarted : acc) + Finished -> reverse (toInput (getStarted welcomeHint) : acc) + PreviouslyOnboarded -> reverse (toInput (getStarted welcomeHint) : acc) toInput :: P.Pretty P.ColorText -> Either Event Input toInput pretty = @@ -110,9 +111,12 @@ authorSuggestion = P.wrap $ P.blue "https://www.unison-lang.org/learn/tooling/configuration/" ] -getStarted :: P.Pretty P.ColorText -getStarted = +getStarted :: Bool -> P.Pretty P.ColorText +getStarted welcomeHint = P.wrap "📚 Read the official docs at https://www.unison-lang.org/learn/" <> P.newline <> P.newline - <> P.wrap "Type 'project.create' to get started." + <> P.wrap (if welcomeHint + then "Hint: type 'projects' to list all your projects" + else "Type 'project.create' to get started.") + diff --git a/unison-cli/unison/Main.hs b/unison-cli/unison/Main.hs index 4a62847909..3e3002df5f 100644 --- a/unison-cli/unison/Main.hs +++ b/unison-cli/unison/Main.hs @@ -478,13 +478,13 @@ launch :: (Path.Absolute -> STM ()) -> CommandLine.ShouldWatchFiles -> IO () -launch dir config runtime sbRuntime codebase inputs serverBaseUrl mayStartingPath initResult notifyRootChange notifyPathChange shouldWatchFiles = +launch dir config runtime sbRuntime codebase inputs serverBaseUrl mayStartingPath initResult notifyRootChange notifyPathChange shouldWatchFiles = do + welcomeHint <- Codebase.runTransaction codebase Queries.doProjectsExist let isNewCodebase = case initResult of CreatedCodebase -> NewlyCreatedCodebase OpenedCodebase -> PreviouslyCreatedCodebase - (ucmVersion, _date) = Version.gitDescribe - welcome = Welcome.welcome isNewCodebase ucmVersion + welcome = Welcome.welcome isNewCodebase ucmVersion welcomeHint in CommandLine.main dir welcome From 0657d9d571ea9d7fbedce50c6ff636f24c08c0ca Mon Sep 17 00:00:00 2001 From: sixfourtwelve Date: Tue, 5 Sep 2023 23:06:22 +0100 Subject: [PATCH 24/28] make field name more meaningful --- unison-cli/src/Unison/CommandLine/Welcome.hs | 16 ++++++++-------- unison-cli/unison/Main.hs | 4 ++-- 2 files changed, 10 insertions(+), 10 deletions(-) diff --git a/unison-cli/src/Unison/CommandLine/Welcome.hs b/unison-cli/src/Unison/CommandLine/Welcome.hs index 5418f0524d..7510ac5133 100644 --- a/unison-cli/src/Unison/CommandLine/Welcome.hs +++ b/unison-cli/src/Unison/CommandLine/Welcome.hs @@ -15,7 +15,7 @@ import Prelude hiding (readFile, writeFile) data Welcome = Welcome { onboarding :: Onboarding, -- Onboarding States unisonVersion :: Text, - welcomeHint :: Bool + showWelcomeHint :: Bool } -- Previously Created is different from Previously Onboarded because a user can @@ -36,11 +36,11 @@ data Onboarding deriving (Show, Eq) welcome :: CodebaseInitStatus -> Text -> Bool -> Welcome -welcome initStatus unisonVersion welcomeHint = - Welcome (Init initStatus) unisonVersion welcomeHint +welcome initStatus unisonVersion showWelcomeHint = + Welcome (Init initStatus) unisonVersion showWelcomeHint run :: Welcome -> [Either Event Input] -run Welcome {onboarding = onboarding, unisonVersion = version, welcomeHint = welcomeHint} = do +run Welcome {onboarding = onboarding, unisonVersion = version, showWelcomeHint = showWelcomeHint} = do go onboarding [] where go :: Onboarding -> [Either Event Input] -> [Either Event Input] @@ -59,8 +59,8 @@ run Welcome {onboarding = onboarding, unisonVersion = version, welcomeHint = wel where authorMsg = toInput authorSuggestion -- These are our two terminal Welcome conditions, at the end we reverse the order of the desired input commands otherwise they come out backwards - Finished -> reverse (toInput (getStarted welcomeHint) : acc) - PreviouslyOnboarded -> reverse (toInput (getStarted welcomeHint) : acc) + Finished -> reverse (toInput (getStarted showWelcomeHint) : acc) + PreviouslyOnboarded -> reverse (toInput (getStarted showWelcomeHint) : acc) toInput :: P.Pretty P.ColorText -> Either Event Input toInput pretty = @@ -112,11 +112,11 @@ authorSuggestion = ] getStarted :: Bool -> P.Pretty P.ColorText -getStarted welcomeHint = +getStarted showWelcomeHint = P.wrap "📚 Read the official docs at https://www.unison-lang.org/learn/" <> P.newline <> P.newline - <> P.wrap (if welcomeHint + <> P.wrap (if showWelcomeHint then "Hint: type 'projects' to list all your projects" else "Type 'project.create' to get started.") diff --git a/unison-cli/unison/Main.hs b/unison-cli/unison/Main.hs index 3e3002df5f..d439e256f6 100644 --- a/unison-cli/unison/Main.hs +++ b/unison-cli/unison/Main.hs @@ -479,12 +479,12 @@ launch :: CommandLine.ShouldWatchFiles -> IO () launch dir config runtime sbRuntime codebase inputs serverBaseUrl mayStartingPath initResult notifyRootChange notifyPathChange shouldWatchFiles = do - welcomeHint <- Codebase.runTransaction codebase Queries.doProjectsExist + showWelcomeHint <- Codebase.runTransaction codebase Queries.doProjectsExist let isNewCodebase = case initResult of CreatedCodebase -> NewlyCreatedCodebase OpenedCodebase -> PreviouslyCreatedCodebase (ucmVersion, _date) = Version.gitDescribe - welcome = Welcome.welcome isNewCodebase ucmVersion welcomeHint + welcome = Welcome.welcome isNewCodebase ucmVersion showWelcomeHint in CommandLine.main dir welcome From d73840fa349a177625c600d5f75a6e46dfdf8656 Mon Sep 17 00:00:00 2001 From: sixfourtwelve Date: Wed, 6 Sep 2023 06:22:42 +0100 Subject: [PATCH 25/28] fix typo at end of message --- unison-cli/src/Unison/CommandLine/Welcome.hs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/unison-cli/src/Unison/CommandLine/Welcome.hs b/unison-cli/src/Unison/CommandLine/Welcome.hs index 7510ac5133..0653cb9ced 100644 --- a/unison-cli/src/Unison/CommandLine/Welcome.hs +++ b/unison-cli/src/Unison/CommandLine/Welcome.hs @@ -117,6 +117,6 @@ getStarted showWelcomeHint = <> P.newline <> P.newline <> P.wrap (if showWelcomeHint - then "Hint: type 'projects' to list all your projects" + then "Hint: type 'projects' to list all your projects." else "Type 'project.create' to get started.") From f96bf79d3d0635e3725f506f3b36165f5bf5d0bc Mon Sep 17 00:00:00 2001 From: ethan Date: Wed, 6 Sep 2023 13:03:41 +0100 Subject: [PATCH 26/28] pat myself on the back --- CONTRIBUTORS.markdown | 1 + 1 file changed, 1 insertion(+) diff --git a/CONTRIBUTORS.markdown b/CONTRIBUTORS.markdown index 52881ca1e8..20107de877 100644 --- a/CONTRIBUTORS.markdown +++ b/CONTRIBUTORS.markdown @@ -77,3 +77,4 @@ The format for this list: name, GitHub handle * Chris Krycho (@chriskrycho) * Hatim Khambati (@hatimkhambati26) * Kyle Goetz (@kylegoetz) +* Ethan Morgan (@sixfourtwelve) From 7c3df64c8ddca530239a7a3460e5cdbd0c869a3a Mon Sep 17 00:00:00 2001 From: Chris Penner Date: Wed, 6 Sep 2023 10:49:05 -0700 Subject: [PATCH 27/28] Relax hash-mismatch errors if the remote head is what we want --- .../Codebase/Editor/HandleInput/Push.hs | 28 +++++++++++-------- 1 file changed, 17 insertions(+), 11 deletions(-) diff --git a/unison-cli/src/Unison/Codebase/Editor/HandleInput/Push.hs b/unison-cli/src/Unison/Codebase/Editor/HandleInput/Push.hs index 0ee10258d3..6fce387a15 100644 --- a/unison-cli/src/Unison/Codebase/Editor/HandleInput/Push.hs +++ b/unison-cli/src/Unison/Codebase/Editor/HandleInput/Push.hs @@ -749,7 +749,24 @@ makeSetHeadAfterUploadAction force pushing localBranchHead remoteBranch = do branchOldCausalHash = Just remoteBranchHead, branchNewCausalHash = localBranchHead } + let onSuccess = + case pushing of + PushingLooseCode -> pure () + PushingProjectBranch (ProjectAndBranch localProject localBranch) -> do + Cli.runTransaction do + Queries.ensureBranchRemoteMapping + (localProject ^. #projectId) + (localBranch ^. #branchId) + (remoteBranch ^. #projectId) + Share.hardCodedUri + (remoteBranch ^. #branchId) Share.setProjectBranchHead request >>= \case + Share.SetProjectBranchHeadResponseSuccess -> onSuccess + -- Sometimes a different request gets through in between checking the remote head and + -- executing the check-and-set push, if it managed to set the head to what we wanted + -- then the goal was achieved and we can consider it a success. + Share.SetProjectBranchHeadResponseExpectedCausalHashMismatch _expected actual + | actual == localBranchHead -> onSuccess Share.SetProjectBranchHeadResponseExpectedCausalHashMismatch _expected _actual -> Cli.returnEarly (RemoteProjectBranchHeadMismatch Share.hardCodedUri remoteProjectAndBranchNames) Share.SetProjectBranchHeadResponseNotFound -> do @@ -758,17 +775,6 @@ makeSetHeadAfterUploadAction force pushing localBranchHead remoteBranch = do Cli.returnEarly (Output.RemoteProjectReleaseIsDeprecated Share.hardCodedUri remoteProjectAndBranchNames) Share.SetProjectBranchHeadResponsePublishedReleaseIsImmutable -> do Cli.returnEarly (Output.RemoteProjectPublishedReleaseCannotBeChanged Share.hardCodedUri remoteProjectAndBranchNames) - Share.SetProjectBranchHeadResponseSuccess -> do - case pushing of - PushingLooseCode -> pure () - PushingProjectBranch (ProjectAndBranch localProject localBranch) -> do - Cli.runTransaction do - Queries.ensureBranchRemoteMapping - (localProject ^. #projectId) - (localBranch ^. #branchId) - (remoteBranch ^. #projectId) - Share.hardCodedUri - (remoteBranch ^. #branchId) where remoteBranchHead = Share.API.hashJWTHash (remoteBranch ^. #branchHead) From 2c1ab8b0456e1f1654b99912fce58ac7529a01e5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?ethan=20=EF=A3=BF?= Date: Wed, 6 Sep 2023 19:54:30 +0100 Subject: [PATCH 28/28] Update unison-cli/src/Unison/CommandLine/Welcome.hs Co-authored-by: Arya Irani <538571+aryairani@users.noreply.github.com> --- unison-cli/src/Unison/CommandLine/Welcome.hs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/unison-cli/src/Unison/CommandLine/Welcome.hs b/unison-cli/src/Unison/CommandLine/Welcome.hs index 0653cb9ced..6e8734c868 100644 --- a/unison-cli/src/Unison/CommandLine/Welcome.hs +++ b/unison-cli/src/Unison/CommandLine/Welcome.hs @@ -117,6 +117,6 @@ getStarted showWelcomeHint = <> P.newline <> P.newline <> P.wrap (if showWelcomeHint - then "Hint: type 'projects' to list all your projects." + then "Hint: Type 'projects' to list all your projects, or 'project.create' to start something new." else "Type 'project.create' to get started.")