forked from fscheck/FsCheck
-
Notifications
You must be signed in to change notification settings - Fork 0
/
build.fsx
443 lines (368 loc) · 16.2 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
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
// --------------------------------------------------------------------------------------
// FAKE build script
// --------------------------------------------------------------------------------------
#r "paket: groupref Build //"
#load "./.fake/build.fsx/intellisense.fsx"
open System
open System.IO
open Fake.Api
open Fake.Core
open Fake.Core.TargetOperators
open Fake.BuildServer
open Fake.DotNet
open Fake.DotNet.Testing
open Fake.IO
open Fake.IO.FileSystemOperators
open Fake.IO.Globbing.Operators
open Fake.Tools.Git
// Information about each project is used
// - for version and project name in generated AssemblyInfo file
// - by the generated NuGet package
// - to run tests and to publish documentation on GitHub gh-pages
// - for documentation, you also need to edit info in "docs/tools/generate.fsx"
type ProjectInfo =
{ /// The name of the project
/// (used by attributes in AssemblyInfo, name of a NuGet package and directory in 'src')
Name : string
/// Short summary of the project
/// (used as description in AssemblyInfo and as a short summary for NuGet package)
Summary : string
}
//File that contains the release notes.
let releaseNotes = "FsCheck Release Notes.md"
/// Solution or project files to be built during the building process
let solution = "FsCheck.sln"
/// Pattern specifying assemblies to be tested
let testAssemblies = "tests/**/bin/Release/net452/*.Test.dll"
// Git configuration (used for publishing documentation in gh-pages branch)
// The profile where the project is posted
let gitOwner = "fscheck"
let gitHome = sprintf "ssh://github.com/%s" gitOwner
// gitraw location - used for source linking
let gitRaw = Environment.environVarOrDefault "gitRaw" "https://raw.github.com/fscheck"
// The name of the project on GitHub
let gitName = "FsCheck"
// Read additional information from the release notes document
Environment.CurrentDirectory <- __SOURCE_DIRECTORY__
let release = ReleaseNotes.load releaseNotes
let isAppVeyorBuild = BuildServer.buildServer = BuildServer.AppVeyor
let buildDate = DateTime.UtcNow
let buildVersion =
let isVersionTag (tag:string) = Version.TryParse tag |> fst
let hasRepoVersionTag = isAppVeyorBuild && AppVeyor.Environment.RepoTag && isVersionTag AppVeyor.Environment.RepoTagName
let assemblyVersion = if hasRepoVersionTag then AppVeyor.Environment.RepoTagName else release.NugetVersion
if isAppVeyorBuild then sprintf "%s-b%s" assemblyVersion AppVeyor.Environment.BuildNumber
else assemblyVersion
let packages =
[
{ Name = "FsCheck"
Summary = "FsCheck is a tool for testing .NET programs automatically using randomly generated test cases."
}
{ Name = "FsCheck.NUnit"
Summary = "Integrates FsCheck with NUnit"
}
{ Name = "FsCheck.Xunit"
Summary = "Integrates FsCheck with xUnit.NET"
}
]
Target.create "BuildVersion" (fun _ ->
Shell.Exec("appveyor", sprintf "UpdateBuild -Version \"%s\"" buildVersion) |> ignore
)
// Generate assembly info files with the right version & up-to-date information
Target.create "AssemblyInfo" (fun _ ->
packages |> Seq.iter (fun package ->
let fileName = "src/" + package.Name + "/AssemblyInfo.fs"
AssemblyInfoFile.createFSharp fileName
([AssemblyInfo.Title package.Name
AssemblyInfo.Product package.Name
AssemblyInfo.Description package.Summary
AssemblyInfo.Version release.AssemblyVersion
AssemblyInfo.FileVersion release.AssemblyVersion
] @ (if package.Name = "FsCheck" || package.Name = "FsCheck.Xunit"
then [Fake.DotNet.AssemblyInfo.InternalsVisibleTo("FsCheck.Test")] else []))
)
)
// --------------------------------------------------------------------------------------
// Clean build results
Target.create "Clean" (fun _ ->
Shell.cleanDirs ["bin"; "temp"]
)
Target.create "CleanDocs" (fun _ ->
Shell.cleanDirs ["docs/output"]
)
// --------------------------------------------------------------------------------------
// Build library & test project
Target.create "Build" (fun _ ->
DotNet.restore id solution
DotNet.build (fun opt -> { opt with Configuration = DotNet.BuildConfiguration.Release }) solution
)
// --------------------------------------------------------------------------------------
// Run the unit tests using test runner
Target.create "RunTests" (fun _ ->
!! testAssemblies
|> XUnit2.run (fun p ->
{ p with
//ToolPath = "packages/build/xunit.runner.console/tools/xunit.console.exe"
//The NoAppDomain setting requires care.
//On mono, it needs to be true otherwise xunit won't work due to a Mono bug.
//On .NET, it needs to be false otherwise Unquote won't work because it won't be able to load the FsCheck assembly.
NoAppDomain = Environment.isMono
ShadowCopy = false })
)
// --------------------------------------------------------------------------------------
// Build a NuGet package
Target.create "PaketPack" (fun _ ->
Paket.pack (fun p ->
{ p with
OutputPath = "bin"
Version = buildVersion
ReleaseNotes = String.toLines release.Notes
})
)
Target.create "PaketPush" (fun _ ->
Paket.push (fun p ->
{ p with
WorkingDir = "bin"
})
)
// --------------------------------------------------------------------------------------
// Generate the documentation
// Paths with template/source/output locations
let bin = __SOURCE_DIRECTORY__ @@ "src/FsCheck.Xunit/bin/Release/netstandard2.0" //might not work in the future
let content = __SOURCE_DIRECTORY__ @@ "docs/content"
let output = __SOURCE_DIRECTORY__ @@ "docs/output"
let files = __SOURCE_DIRECTORY__ @@ "docs/files"
let templates = __SOURCE_DIRECTORY__ @@ "docs/tools/templates"
let formatting = __SOURCE_DIRECTORY__ @@ "packages/build/FSharp.Formatting/"
let docTemplate = formatting @@ "templates/docpage.cshtml"
let github_release_user = Environment.environVarOrDefault "github_release_user" gitOwner
let githubLink = sprintf "https://github.com/%s/%s" github_release_user gitName
// Specify more information about your project
let info =
[ "project-name", "FsCheck"
"project-author", "Kurt Schelfthout and contributors"
"project-summary", "FsCheck is a tool for testing .NET programs automatically using randomly generated test cases."
"project-github", githubLink
"project-nuget", "http://nuget.org/packages/FsCheck" ]
// Web site location for the generated documentation
let website = "/FsCheck"
let root = website
// Binaries that have XML documentation (in a corresponding generated XML file)
let referenceBinaries = [ "FsCheck.dll" ]
let layoutRootsAll = new System.Collections.Generic.Dictionary<string, string list>()
layoutRootsAll.Add("en",[ templates;
formatting @@ "templates"
formatting @@ "templates/reference" ])
Target.create "ReferenceDocs" (fun _ ->
Directory.ensure (output @@ "reference")
let binaries () =
let manuallyAdded =
referenceBinaries
|> List.map (fun b -> bin @@ b)
let conventionBased = []
//DirectoryInfo.getSubDirectories <| DirectoryInfo bin
//|> Array.collect (fun d ->
// let name, dInfo =
// let net45Bin =
// DirectoryInfo.getSubDirectories d |> Array.filter(fun x -> x.FullName.ToLower().Contains("net45"))
// let net47Bin =
// DirectoryInfo.getSubDirectories d |> Array.filter(fun x -> x.FullName.ToLower().Contains("net47"))
// if net45Bin.Length > 0 then
// d.Name, net45Bin.[0]
// else
// d.Name, net47Bin.[0]
// dInfo.GetFiles()
// |> Array.filter (fun x ->
// x.Name.ToLower() = (sprintf "%s.dll" name).ToLower())
// |> Array.map (fun x -> x.FullName)
// )
//|> List.ofArray
conventionBased @ manuallyAdded
binaries()
|> FSFormatting.createDocsForDlls (fun args ->
{ args with
OutputDirectory = output @@ "reference"
LayoutRoots = layoutRootsAll.["en"]
ProjectParameters = ("root", root)::info
SourceRepository = githubLink @@ "tree/master" }
)
)
let copyFiles () =
Shell.copyRecursive files output true
|> Trace.logItems "Copying file: "
Directory.ensure (output @@ "content")
Shell.copyRecursive (formatting @@ "styles") (output @@ "content") true
|> Trace.logItems "Copying styles and scripts: "
/// Specifies the fsformatting executable
let toolPath() =
Fake.Core.ProcessUtils.tryFindLocalTool "FSFORMATTING" "fsformatting.exe"
[(Directory.GetCurrentDirectory() @@ "packages" @@ "build" @@ "FSharp.Formatting.CommandTool" @@ "tools")]
|> Option.get
/// Runs fsformatting.exe with the given command in the given repository directory.
let private run toolPath command =
let result = CreateProcess.fromRawCommandLine toolPath command
|> CreateProcess.withFramework
|> CreateProcess.redirectOutput
|> Proc.run
if result.ExitCode <> 0 then
failwithf "%s %s failed with exit code %i. StdOut: %s ErrOut: %s" toolPath command result.ExitCode result.Result.Output result.Result.Error
type LiterateArguments =
{ ToolPath : string
Source : string
OutputDirectory : string
Template : string
ProjectParameters : (string * string) list
LayoutRoots : string list }
let createDocs p =
let defaultLiterateArguments =
{ ToolPath = toolPath()
Source = ""
OutputDirectory = ""
Template = ""
ProjectParameters = []
LayoutRoots = [] }
let arguments = (p:LiterateArguments->LiterateArguments) defaultLiterateArguments
let layoutroots =
if arguments.LayoutRoots.IsEmpty then []
else [ "--layoutRoots" ] @ arguments.LayoutRoots
let source = arguments.Source
let template = arguments.Template
let outputDir = arguments.OutputDirectory
let command =
arguments.ProjectParameters
|> Seq.map (fun (k, v) -> [ k; v ])
|> Seq.concat
|> Seq.append
(["literate"; "--processDirectory" ] @ layoutroots @ [ "--inputDirectory"; source; "--templateFile"; template;
"--fsieval"; "--outputDirectory"; outputDir; "--replacements" ])
|> Seq.map (fun s ->
if s.StartsWith "\"" then s
else sprintf "\"%s\"" s)
|> String.separated " "
run arguments.ToolPath command
printfn "Successfully generated docs for %s" source
Target.create "Docs" (fun _ ->
//File.delete "docs/content/release-notes.md"
//Shell.copyFile "docs/content/" "RELEASE_NOTES.md"
//Shell.rename "docs/content/release-notes.md" "docsrc/content/RELEASE_NOTES.md"
//File.delete "docsrc/content/license.md"
//Shell.copyFile "docsrc/content/" "LICENSE.txt"
//Shell.rename "docsrc/content/license.md" "docsrc/content/LICENSE.txt"
DirectoryInfo.getSubDirectories (DirectoryInfo.ofPath templates)
|> Seq.iter (fun d ->
let name = d.Name
if name.Length = 2 || name.Length = 3 then
layoutRootsAll.Add(
name, [templates @@ name
formatting @@ "templates"
formatting @@ "templates/reference" ]))
copyFiles ()
for dir in [ content; ] do
let langSpecificPath(lang, path:string) =
path.Split([|'/'; '\\'|], System.StringSplitOptions.RemoveEmptyEntries)
|> Array.exists(fun i -> i = lang)
let layoutRoots =
let key = layoutRootsAll.Keys |> Seq.tryFind (fun i -> langSpecificPath(i, dir))
match key with
| Some lang -> layoutRootsAll.[lang]
| None -> layoutRootsAll.["en"] // "en" is the default language
createDocs (fun args ->
{ args with
Source = content
OutputDirectory = output
LayoutRoots = layoutRoots
ProjectParameters = ("root", root)::info
Template = docTemplate } )
)
//// --------------------------------------------------------------------------------------
//// Generate the documentation
//let generateHelp' fail debug =
// let args =
// if debug then ["--define:HELP"]
// else ["--define:RELEASE"; "--define:HELP"]
// if Fake.FSIHelper.executeFSIWithArgs "docs/tools" "generate.fsx" args [] then
// Trace.traceImportant "Help generated"
// else
// if fail then
// failwith "generating help documentation failed"
// else
// Trace.traceImportant "generating help documentation failed"
//let generateHelp fail =
// generateHelp' fail true
//Target.create "KeepRunning" (fun _ ->
// use watcher = new FileSystemWatcher(DirectoryInfo("docs/content").FullName,"*.fsx")
// watcher.EnableRaisingEvents <- true
// watcher.Changed.Add(fun e -> Trace.trace (sprintf "%A %A" e.Name e.ChangeType); generateHelp false)
// watcher.Created.Add(fun e -> Trace.trace (sprintf "%A %A" e.Name e.ChangeType); generateHelp false)
// watcher.Renamed.Add(fun e -> Trace.trace (sprintf "%A %A" e.Name e.ChangeType); generateHelp false)
// //watcher.Deleted.Add(fun e -> trace (sprintf "%A %A" e.Name e.ChangeType); generateHelp false)
// Trace.traceImportant "Waiting for help edits. Press any key to stop."
// System.Console.ReadKey() |> ignore
// watcher.EnableRaisingEvents <- false
// watcher.Dispose()
//)
//Target.create "GenerateDocs" (fun _ ->
// Fake.FSIHelper.executeFSIWithArgs "docs/tools" "generate.fsx" ["--define:RELEASE"; "--define:HELP"; "--define:REFERENCE"] [] |> ignore
//)
//Target.create "GenerateDocsJa" (fun _ ->
// Fake.FSIHelper.executeFSIWithArgs "docs/tools" "generate.ja.fsx" ["--define:RELEASE"] [] |> ignore
//)
// --------------------------------------------------------------------------------------
// Release Scripts
Target.create "ReleaseDocs" (fun _ ->
let tempDocsDir = "temp/gh-pages"
Shell.cleanDir tempDocsDir
Repository.cloneSingleBranch "" ("[email protected]:fscheck/FsCheck.git") "gh-pages" tempDocsDir
Repository.fullclean tempDocsDir
Shell.copyRecursive "docs/output" tempDocsDir true |> Trace.tracefn "%A"
Staging.stageAll tempDocsDir
Commit.exec tempDocsDir (sprintf "Update generated documentation for version %s" buildVersion)
Branches.push tempDocsDir
)
Target.create "Release" (fun _ ->
let user = Environment.environVarOrDefault "github-user" (UserInput.getUserInput "Username: ")
let pw = Environment.environVarOrDefault "github-pw" (UserInput.getUserPassword "Password: ")
let remote =
CommandHelper.getGitResult "" "remote -v"
|> Seq.filter (fun (s: string) -> s.EndsWith("(push)"))
|> Seq.tryFind (fun (s: string) -> s.Contains(gitOwner + "/" + gitName))
|> function None -> gitHome + "/" + gitName | Some (s: string) -> s.Split().[0]
Staging.stageAll ""
Commit.exec "" (sprintf "Bump version to %s" release.NugetVersion)
Branches.pushBranch "" remote (Information.getBranchName "")
Branches.tag "" release.NugetVersion
Branches.pushTag "" remote release.NugetVersion
// release on github
GitHub.createClient user pw
|> GitHub.draftNewRelease gitOwner gitName release.NugetVersion (release.SemVer.PreRelease <> None) release.Notes
// to upload a file: |> GitHub.uploadFiles "PATH_TO_FILE"
|> GitHub.publishDraft
|> Async.RunSynchronously
)
// --------------------------------------------------------------------------------------
// Run all targets by default. Invoke 'build <Target>' to override
Target.create "CI" ignore
Target.create "Tests" ignore
"Clean"
=?> ("BuildVersion", isAppVeyorBuild)
==> "AssemblyInfo"
==> "Build"
==> "RunTests"
==> "Tests"
"Build"
==> "ReferenceDocs"
"CleanDocs"
==> "ReferenceDocs"
==> "Docs"
=?> ("ReleaseDocs", BuildServer.isLocalBuild)
==> "Release"
"Tests"
==> "PaketPack"
==> "PaketPush"
==> "Release"
"Docs"
==> "CI"
"Tests"
==> "PaketPack"
==> "CI"
Target.runOrDefault "RunTests"