-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathbuild.fsx
268 lines (218 loc) · 10.6 KB
/
build.fsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
// --------------------------------------------------------------------------------------
// FAKE build script
// --------------------------------------------------------------------------------------
#I @".\packages\Newtonsoft.Json\lib\netstandard1.3"
#r @"packages/build/FAKE/tools/FakeLib.dll"
#r @"packages/build/Microsoft.Rest.ClientRuntime.Azure/lib/netstandard1.4/Microsoft.Rest.ClientRuntime.Azure.dll"
#load @"src\scripts\importdata.fsx"
@".paket\load\netstandard2.0\Build\build.group.fsx"
@"paket-files\build\CompositionalIT\fshelpers\src\FsHelpers\ArmHelper\ArmHelper.fs"
open Cit.Helpers.Arm
open Cit.Helpers.Arm.Parameters
open Fake
open System
open System.IO
open Microsoft.IdentityModel.Clients.ActiveDirectory
open Microsoft.Azure.Management.ResourceManager.Fluent
open Microsoft.Azure.Management.ResourceManager.Fluent.Core
let project = "Property Mapper"
let summary = "A project to illustrate use of Azure Search to map UK property sales using SAFE."
let description = summary
let clientPath = FullName "./src/Client"
let serverPath = FullName "./src/Server/"
let dotnetcliVersion = DotNetCli.getVersion()
let mutable dotnetExePath = "dotnet"
let deployDir = "./deploy"
// --------------------------------------------------------------------------------------
// END TODO: The rest of the file includes standard build steps
// --------------------------------------------------------------------------------------
let run cmd args dir =
if execProcess (fun info ->
info.FileName <- cmd
if not (String.IsNullOrWhiteSpace dir) then
info.WorkingDirectory <- dir
info.Arguments <- args
) System.TimeSpan.MaxValue |> not then
failwithf "Error while running '%s' with args: %s" cmd args
let runDotnet workingDir args =
let result =
ExecProcess (fun info ->
info.FileName <- dotnetExePath
info.WorkingDirectory <- workingDir
info.Arguments <- args) TimeSpan.MaxValue
if result <> 0 then failwithf "dotnet %s failed" args
let platformTool tool winTool =
let tool = if isUnix then tool else winTool
tool
|> ProcessHelper.tryFindFileOnPath
|> function Some t -> t | _ -> failwithf "%s not found" tool
let nodeTool = platformTool "node" "node.exe"
let npmTool = platformTool "npm" "npm.cmd"
let yarnTool = platformTool "yarn" "yarn.cmd"
do if not isWindows then
// We have to set the FrameworkPathOverride so that dotnet sdk invocations know
// where to look for full-framework base class libraries
let mono = platformTool "mono" "mono"
let frameworkPath = IO.Path.GetDirectoryName(mono) </> ".." </> "lib" </> "mono" </> "4.5"
setEnvironVar "FrameworkPathOverride" frameworkPath
// --------------------------------------------------------------------------------------
// Clean build results
Target "Clean" (fun _ ->
!!"src/**/bin" |> CleanDirs
!! "src/**/obj/*.nuspec" |> DeleteFiles
CleanDirs ["bin"; "temp"; "docs/output"; deployDir; Path.Combine(clientPath,"public/bundle")])
Target "InstallDotNetCore" (fun _ -> dotnetExePath <- DotNetCli.InstallDotNetSDK dotnetcliVersion)
// --------------------------------------------------------------------------------------
// Build library
Target "BuildServer" (fun _ -> runDotnet serverPath "build")
Target "InstallClient" (fun _ ->
printfn "Node version:"
run nodeTool "--version" __SOURCE_DIRECTORY__
printfn "Yarn version:"
run yarnTool "--version" __SOURCE_DIRECTORY__
run yarnTool "install --frozen-lockfile" __SOURCE_DIRECTORY__)
Target "BuildClient" (fun _ ->
runDotnet clientPath "restore"
runDotnet clientPath "fable webpack -- -p")
// --------------------------------------------------------------------------------------
// Azure Deployment
Target "DeployArmTemplate" <| fun _ ->
let armTemplate = @"src\arm-template.json"
let environment = getBuildParamOrDefault "environment" (Guid.NewGuid().ToString().ToLower().Split '-' |> Array.head)
let resourceGroupName = sprintf "safe-property-mapper"
tracefn "Deploying template '%s' to resource group '%s'..." armTemplate resourceGroupName
let deployment =
{ DeploymentName = "FAKE-PropertyMapper-Deploy"
ResourceGroup = ResourceGroupType.New(resourceGroupName, Microsoft.Azure.Management.ResourceManager.Fluent.Core.Region.EuropeWest)
ArmTemplate = File.ReadAllText armTemplate
Parameters =
[ "environment", environment
"searchSize", getBuildParam "searchSize"
"webServerSize", getBuildParam "webServerSize"
"alwaysOn", getBuildParam "alwaysOn" ]
|> List.choose(fun (k, v) -> if String.IsNullOrWhiteSpace v then None else Some (k, ArmString v))
|> Parameters.Simple
DeploymentMode = Incremental }
let authCtx =
match getBuildParam "subscriptionId", (getBuildParam "clientId", getBuildParam "clientSecret", getBuildParam "tenantId") with
| "", _ -> failwith "No subscription id supplied!"
| subscriptionId, ("", _, _ | _, "", _ | _, _, "")->
let client = AuthenticationContext "https://login.microsoftonline.com/common"
async {
let! deviceResult = client.AcquireDeviceCodeAsync("https://management.core.windows.net/", "84edda74-cb5b-49e3-99b3-82d1c419e651") |> Async.AwaitTask
tracefn "%s" deviceResult.Message
let! tokenResult = client.AcquireTokenByDeviceCodeAsync(deviceResult) |> Async.AwaitTask
let client = RestClient.Configure().WithEnvironment(AzureEnvironment.AzureGlobalCloud).WithCredentials(Microsoft.Rest.TokenCredentials(tokenResult.AccessToken)).Build()
return ResourceManager.Authenticate(client).WithSubscription subscriptionId |> AuthenticatedContext } |> Async.RunSynchronously
| subscriptionId, (clientId, clientSecret, tenantId) ->
let authCredentials =
{ ClientId = Guid.Parse clientId
ClientSecret = clientSecret
TenantId = Guid.Parse tenantId }
authenticate authCredentials (Guid.Parse subscriptionId)
deployment
|> deployWithProgress authCtx
|> Seq.iter(function
| DeploymentInProgress (state, operations) -> tracefn "State is %s, completed %d operations." state operations
| DeploymentError (statusCode, message) -> traceError <| sprintf "DEPLOYMENT ERROR: %s - '%s'" statusCode message
| DeploymentCompleted _ -> ())
// --------------------------------------------------------------------------------------
// Data Import
open PropertyMapper
open Importdata
let (|Local|Cloud|) (x:string) =
match x.ToLower() with
| "cloud" -> Cloud
| _ -> Local
Target "ImportData" (fun _ ->
let mode = getBuildParam "DataMode"
// Insert postcode / geo lookup
if not (fileExists (FullName "ukpostcodes.csv")) then
log "Downloading transaction data..."
let txns = fetchTransactions 1000
log "Downloading geolocation data..."
let archivePath = "ukpostcodes.zip"
do
use wc = new Net.WebClient()
wc.DownloadFile(Uri "https://www.freemaptools.com/download/full-postcodes/ukpostcodes.zip", archivePath)
archivePath |> Unzip "."
DeleteFile archivePath
let postCodes =
let loadedPostcodes = txns |> Array.Parallel.choose(fun t -> t.Address.PostCode) |> Set
fetchPostcodes (FullName "ukpostcodes.csv")
|> Array.filter(fun (r:Importdata.GeoPostcode) -> loadedPostcodes.Contains r.PostCodeDescription)
let tryFindGeo = postCodes |> Seq.map(fun r -> r.PostCodeDescription, (r.Latitude, r.Longitude)) |> Map.ofSeq |> fun m -> m.TryFind
let insertPostcodeLookup (ConnectionString connectionString) =
postCodes
|> insertPostcodes connectionString
|> Array.collect snd
|> Array.countBy(function FSharp.Azure.StorageTypeProvider.Table.SuccessfulResponse _ -> "Success" | _ -> "Failed")
|> logfn "%A"
log "Now inserting property transactions into search index and creating postcode lookup..."
match mode with
| Local ->
let path = serverPath </> "properties.json"
File.WriteAllText(path, FableJson.toJson txns)
AzureHelper.StartStorageEmulator()
insertPostcodeLookup (ConnectionString "UseDevelopmentStorage=true")
| Cloud ->
let config =
{ AzureSearchServiceName = ""
AzureStorage = ConnectionString ""
AzureSearch = ConnectionString "" }
Search.Azure.Management.initialize config
txns
|> Search.Azure.insertProperties config tryFindGeo
|> fun t -> t.Result
|> logf "%A"
insertPostcodeLookup config.AzureStorage)
// --------------------------------------------------------------------------------------
// Run the Website
let ipAddress = "localhost"
let port = 8080
FinalTarget "KillProcess" (fun _ ->
killProcess "dotnet"
killProcess "dotnet.exe")
Target "Run" (fun _ ->
runDotnet clientPath "restore"
let server = async { runDotnet serverPath "run" }
let fablewatch = async { runDotnet clientPath "fable webpack-dev-server" }
let openBrowser = async {
System.Threading.Thread.Sleep(5000)
Diagnostics.Process.Start("http://"+ ipAddress + sprintf ":%d" port) |> ignore }
Async.Parallel [| server; fablewatch; openBrowser |]
|> Async.RunSynchronously
|> ignore)
Target "BundleClient" (fun _ ->
let result =
ExecProcess (fun info ->
info.FileName <- dotnetExePath
info.WorkingDirectory <- serverPath
info.Arguments <- "publish -c Release -o \"" + FullName deployDir + "\"") TimeSpan.MaxValue
if result <> 0 then failwith "Publish failed"
let clientDir = deployDir </> "client"
let publicDir = clientDir </> "public"
let jsDir = clientDir </> "js"
let cssDir = clientDir </> "css"
let imageDir = clientDir </> "Images"
!! "src/Client/public/**/*.*" |> CopyFiles publicDir
!! "src/Client/js/**/*.*" |> CopyFiles jsDir
!! "src/Client/css/**/*.*" |> CopyFiles cssDir
!! "src/Client/Images/**/*.*" |> CopyFiles imageDir
"src/Client/index.html" |> CopyFile clientDir)
// -------------------------------------------------------------------------------------
Target "Build" DoNothing
Target "All" DoNothing
"Clean"
==> "InstallDotNetCore"
==> "InstallClient"
==> "BuildServer"
==> "BuildClient"
==> "BundleClient"
==> "All"
"BuildClient"
==> "Build"
"InstallClient"
==> "ImportData"
==> "Run"
RunTargetOrDefault "Run"