1
+ module Wasp.Cli.Command.Version.Download
2
+ ( downloadVersion
3
+ , updateWasp
4
+ , getLatestVersionFromGithub
5
+ , isVersionLessThan
6
+ , forceInstallLatest
7
+ , forceInstallSpecific
8
+ ) where
9
+
10
+ import Control.Monad (when )
11
+ import Control.Exception (try , SomeException )
12
+ import Network.HTTP.Simple
13
+ ( httpBS
14
+ , getResponseBody
15
+ , parseRequest
16
+ , setRequestHeader
17
+ , getResponseStatusCode
18
+ , httpNoBody
19
+ , Response
20
+ )
21
+ import qualified Data.ByteString.Char8 as BS
22
+ import System.IO.Temp (withSystemTempDirectory )
23
+ import System.Directory
24
+ ( doesDirectoryExist
25
+ , doesFileExist
26
+ , copyFile
27
+ , removeDirectoryRecursive
28
+ , renameDirectory
29
+ , removeFile
30
+ , executable
31
+ , setPermissions
32
+ , emptyPermissions
33
+ )
34
+ import System.FilePath ((</>) )
35
+ import System.Process (callProcess , callCommand )
36
+ import System.Exit (exitFailure )
37
+ import System.Info (os )
38
+ import Data.Aeson (decode )
39
+ import qualified Data.Aeson.Types as Aeson (parseMaybe , (.:) )
40
+ import qualified Data.ByteString.Lazy.Char8 as LBS
41
+ import Wasp.Cli.Command.Version.Paths
42
+ ( getVersionPaths
43
+ , getVersionFile
44
+ , getWaspRootDir
45
+ , getWaspBinDir
46
+ )
47
+
48
+ -- | Update Wasp to latest version
49
+ updateWasp :: IO ()
50
+ updateWasp = do
51
+ currentVer <- getCurrentReleaseVersion
52
+ latestVer <- getLatestVersionFromGithub
53
+
54
+ when (isVersionLessThan currentVer latestVer) $ do
55
+ putStrLn $ " Updating from ${currentVer} to ${latestVer}..."
56
+ handleDownloadResult =<< try (downloadVersion latestVer)
57
+ updateSystemMetadata latestVer
58
+
59
+ getCurrentReleaseVersion :: IO String
60
+ getCurrentReleaseVersion = getVersionFile " release" >>= readFile
61
+
62
+ handleDownloadResult :: Either SomeException () -> IO ()
63
+ handleDownloadResult (Left e) = do
64
+ putStrLn $ " Update failed: " ++ show e
65
+ exitFailure
66
+ handleDownloadResult (Right _) = return ()
67
+
68
+ updateSystemMetadata :: String -> IO ()
69
+ updateSystemMetadata version = do
70
+ updateMainBinary version
71
+ writeVersionFiles version
72
+
73
+ writeVersionFiles :: String -> IO ()
74
+ writeVersionFiles version = do
75
+ releaseFile <- getVersionFile " release"
76
+ activeFile <- getVersionFile " active"
77
+ writeFile releaseFile version
78
+ writeFile activeFile version
79
+
80
+ -- | Download and install specific version
81
+ downloadVersion :: String -> IO ()
82
+ downloadVersion version = do
83
+ -- Step 1: Check if the GitHub release exists
84
+ versionExists <- checkGitHubRelease version
85
+ if not versionExists
86
+ then do
87
+ putStrLn $ " ❌ Error: Version " ++ version ++ " does not exist. See https://github.com/wasp-lang/wasp/releases for available versions."
88
+ exitFailure
89
+ else do
90
+ putStrLn $ " Starting download..."
91
+
92
+ -- Step 2: Use a temporary directory for download & extraction
93
+ withSystemTempDirectory " wasp-download" $ \ tmpDir -> do
94
+ let archiveFile = tmpDir </> getPlatformString
95
+ downloadArchive version archiveFile
96
+ extractArchive archiveFile tmpDir
97
+ _ <- return tmpDir
98
+
99
+ versionDir <- getVersionDir version
100
+ ensureCleanInstallation versionDir
101
+ renameDirectory tmpDir versionDir
102
+ putStrLn $ " ✅ Wasp version " ++ version ++ " downloaded and activated!"
103
+
104
+ downloadArchive :: String -> FilePath -> IO ()
105
+ downloadArchive version path = do
106
+ let url = " https://github.com/wasp-lang/wasp/releases/download/v"
107
+ ++ version ++ " /" ++ getPlatformString
108
+ putStrLn $ " Downloading from " ++ url
109
+ request <- parseRequest url
110
+ response <- httpBS request
111
+ BS. writeFile path (getResponseBody response)
112
+
113
+ extractArchive :: FilePath -> FilePath -> IO ()
114
+ extractArchive archivePath destDir = do
115
+ putStrLn " Extracting..."
116
+ callProcess " tar" [" -xzf" , archivePath, " -C" , destDir]
117
+ removeFile archivePath
118
+
119
+ ensureCleanInstallation :: FilePath -> IO ()
120
+ ensureCleanInstallation path = do
121
+ exists <- doesDirectoryExist path
122
+ when exists $ do
123
+ putStrLn " Removing existing installation..."
124
+ removeDirectoryRecursive path
125
+
126
+ getVersionDir :: String -> IO FilePath
127
+ getVersionDir version = (</> version) <$> getWaspRootDir
128
+
129
+ -- | Get latest release version from GitHub
130
+ getLatestVersionFromGithub :: IO String
131
+ getLatestVersionFromGithub = do
132
+ response <- httpBS =<< setRequestHeader " User-Agent" [" wasp-cli" ]
133
+ <$> parseRequest " https://api.github.com/repos/wasp-lang/wasp/releases/latest"
134
+ case decodeResponse (getResponseBody response) of
135
+ Just version -> return $ drop 1 version -- Remove 'v' prefix
136
+ Nothing -> error " Failed to parse GitHub response"
137
+
138
+ checkGitHubRelease :: String -> IO Bool
139
+ checkGitHubRelease version = do
140
+ let url = " https://github.com/wasp-lang/wasp/releases/download/v" ++ version ++ " /"
141
+ request <- parseRequest url
142
+ result <- try (httpNoBody request) :: IO (Either SomeException (Response () ))
143
+ case result of
144
+ Right response ->
145
+ let statusCode = getResponseStatusCode response
146
+ in return (statusCode == 200 )
147
+ Left _ -> return False
148
+
149
+ decodeResponse :: BS. ByteString -> Maybe String
150
+ decodeResponse resp = do
151
+ release <- decode (LBS. fromStrict resp)
152
+ Aeson. parseMaybe (Aeson. .: " tag_name" ) release
153
+
154
+ -- Platform-specific configuration
155
+ getPlatformString :: String
156
+ getPlatformString = case os of
157
+ " darwin" -> " wasp-macos-x86_64.tar.gz"
158
+ " linux" -> " wasp-linux-x86_64.tar.gz"
159
+ _ -> error $ " Unsupported OS: " ++ os
160
+
161
+ -- | Create or update the wrapper script
162
+ updateWrapperScript :: FilePath -> FilePath -> FilePath -> IO ()
163
+ updateWrapperScript name binaryPath dataPath = do
164
+ binDir <- getWaspBinDir
165
+ let wrapperPath = binDir </> name
166
+ wrapperContent = unlines
167
+ [ " #!/usr/bin/env bash"
168
+ , " waspc_datadir=" ++ dataPath ++ " " ++ binaryPath ++ " \" $@\" "
169
+ ]
170
+ writeFile wrapperPath wrapperContent
171
+ setPermissions wrapperPath $ emptyPermissions { executable = True }
172
+
173
+ -- | Update the main wasp binary to a specific version
174
+ updateMainBinary :: String -> IO ()
175
+ updateMainBinary version = do
176
+ (versionBin, dataDir) <- getVersionPaths version
177
+ binDir <- getWaspBinDir
178
+ let mainBinary = binDir </> " wasp"
179
+
180
+ exists <- doesFileExist versionBin
181
+ if exists
182
+ then do
183
+ -- Copy the binary
184
+ copyFile versionBin mainBinary
185
+ setPermissions mainBinary $ emptyPermissions { executable = True }
186
+
187
+ -- Update the wrapper script
188
+ updateWrapperScript " wasp" mainBinary dataDir
189
+ putStrLn " Updated wasp wrapper script"
190
+ else error $ " Version " ++ version ++ " binary not found"
191
+
192
+ -- | Semantic version comparison
193
+ isVersionLessThan :: String -> String -> Bool
194
+ isVersionLessThan a b = case (parseVersion a, parseVersion b) of
195
+ (Just v1, Just v2) -> v1 < v2
196
+ _ -> False
197
+
198
+ parseVersion :: String -> Maybe (Int , Int , Int )
199
+ parseVersion v = case reads v of
200
+ [(major, ' .' : rest1)] -> case reads rest1 of
201
+ [(minor, ' .' : rest2)] -> case reads rest2 of
202
+ [(patch, " " )] -> Just (major, minor, patch)
203
+ _ -> Nothing
204
+ _ -> Nothing
205
+ _ -> Nothing
206
+
207
+
208
+ -- | Force install latest version of Wasp
209
+ forceInstallLatest :: IO ()
210
+ forceInstallLatest = do
211
+ releaseFile <- getVersionFile " release"
212
+ activeFile <- getVersionFile " active"
213
+ doesFileExist releaseFile >>= flip when (removeFile releaseFile)
214
+ doesFileExist activeFile >>= flip when (removeFile activeFile)
215
+ putStrLn " Forcing installation of the latest version of Wasp..."
216
+ callCommand " curl -sSL https://get.wasp-lang.dev/installer.sh | sh -s"
217
+
218
+ -- | Force install a specific version of Wasp
219
+ forceInstallSpecific :: String -> IO ()
220
+ forceInstallSpecific version = do
221
+ releaseFile <- getVersionFile " release"
222
+ activeFile <- getVersionFile " active"
223
+ doesFileExist releaseFile >>= flip when (removeFile releaseFile)
224
+ doesFileExist activeFile >>= flip when (removeFile activeFile)
225
+ putStrLn $ " Forcing installation of Wasp version " ++ version ++ " ..."
226
+ callCommand $ " curl -sSL https://get.wasp-lang.dev/installer.sh | sh -s -- -v " ++ version
0 commit comments