Add linux mono support
Closed this issue · 20 comments
It would be nice if running on mono was supported. I have initially attempted to do so and found several problems:
- there is no build scripts for linux at the moment, but solution can be easily build with monodevelop or xbuild.
- zip package is packaged with windows path separators
\
. So when installed withgauge --install csharp --file ..
it results in bad paths on linux, e.g. a filebin\Gauge.CSharp.Runner.exe
instead of directorybin
andbin/Gauge.CSharp.Runner.exe
file - after fixing above installed paths. There plugin config ,must be updated to something like this
~/.gauge/plugins/csharp/0.7.2/csharp.json
. Unfortunately only full paths work
"run": {
"windows": ["bin\\Gauge.Csharp.Runner.exe", "--start"],
"linux": ["/usr/bin/mono", "<full-path-to-home>.gauge/plugins/csharp/0.7.2/bin/Gauge.CSharp.Runner.exe", "--start"],
"darwin": ["echo", "Coming Soon!"]
},
And with all above changes, the runner still fails. I tried executing from IntegrationTestSample
available in this project.
gauge-csharp/IntegrationTestSample$ gauge --verbose=true specs/
Gauge.CSharp.Runner.GaugeProjectBuilder Building Project: /ide/work/gauge-csharp/IntegrationTestSample/IntegrationTestSample.csproj
/ide/work/gauge-csharp/IntegrationTestSample/IntegrationTestSample.csproj: error : Unhandled exception occured during a build
Gauge.CSharp.Runner.GaugeProjectBuilder Failure
I haven figured out yet what is failing when project build is attempted. It can be compiled in monodevelop.
Question to maintainers and contributors - Is there something else that would be needed for linux/mono support, that you know of?
It seems none of above problems is critical, so I could make a PR once I get this to work.
@tomzo - nice to see this get started.
I have played around briefly, but never got around to make this production ready.
On your points:
there is no build scripts for linux at the moment, but solution can be easily build with monodevelop or xbuild.
I believe the problem you are facing is because of Gauge-CSharp
using Microsoft.Build.Framework
APIs. MSBuild does not work on *nix, and need to see an alternative that is truly cross platform.
The relevant code is in GaugeProjectBuilder
zip package is packaged with windows path separators . So when installed with gauge --install csharp --file .. it results in bad paths on linux, e.g. a file bin\Gauge.CSharp.Runner.exe instead of directory bin and bin/Gauge.CSharp.Runner.exe file
Yep, everything in our build script needs to be made cross platform.
after fixing above installed paths. There plugin config ,must be updated to something like this ~/.gauge/plugins/csharp/0.7.2/csharp.json. Unfortunately only full paths work
Strange, I have mono
in my PATH
(in OSx). Here is an extract from my csharp.json
:
"run": {
"windows": ["bin\\Gauge.Csharp.Runner.exe", "--start"],
"linux": ["echo", "Coming Soon!"],
"darwin": ["/usr/local/bin/mono", "bin/Gauge.Csharp.Runner.exe", "--start"]
},
"init": {
"windows": ["bin\\Gauge.Csharp.Runner.exe", "--init"],
"linux": ["echo", "Coming Soon!"],
"darwin": ["/usr/local/bin/mono", "bin/Gauge.Csharp.Runner.exe", "--init"]
},
I could get a basic execution working by following the below steps:
cd /tmp/bar
gauge --init csharp
xbuild bar.csproj
gauge_custom_build_path=bin/Debug gauge specs
Init Output
INb:bar steam$ gauge --init csharp
create bar
create Properties/AssemblyInfo.cs
create StepImplementation.cs
create packages.config
create bar.csproj
create bar.sln
Installing Nuget Package : FluentAssertions, version: 3.5.0
Installing Nuget Package : Gauge.CSharp.Lib, version: 0.5.3
Done Installing Nuget Package!
create manifest.json
create specs
create specs/example.spec
create env
create env/default
create env/default/default.properties
Successfully initialized the project. Run specifications with "gauge specs/"
Build Output
XBuild Engine Version 12.0
Mono, Version 3.12.0.0
Copyright (C) 2005-2013 Various Mono authors
Build started 27-05-2016 13:06:15.
__________________________________________________
Project "/private/tmp/bar/bar.csproj" (default target(s)):
Target PrepareForBuild:
Configuration: Debug Platform: AnyCPU
Created directory "bin/Debug/"
Created directory "obj/Debug/"
Target CopyFilesMarkedCopyLocal:
Copying file from '/private/tmp/bar/packages/FluentAssertions.3.5.0/lib/net45/FluentAssertions.dll' to '/private/tmp/bar/bin/Debug/FluentAssertions.dll'
Copying file from '/private/tmp/bar/packages/FluentAssertions.3.5.0/lib/net45/FluentAssertions.Core.dll' to '/private/tmp/bar/bin/Debug/FluentAssertions.Core.dll'
Copying file from '/private/tmp/bar/packages/Gauge.CSharp.Lib.0.5.3.0/lib/net45/Gauge.CSharp.Lib.dll' to '/private/tmp/bar/bin/Debug/Gauge.CSharp.Lib.dll'
Target GenerateSatelliteAssemblies:
No input files were specified for target GenerateSatelliteAssemblies, skipping.
Target CoreCompile:
Tool /usr/local/Cellar/mono/3.12.0/lib/mono/4.5/mcs.exe execution started with arguments: /noconfig /debug:full /debug+ /optimize- /out:obj/Debug/bar.dll StepImplementation.cs Properties/AssemblyInfo.cs obj/Debug/.NETFramework,Version=v4.5.AssemblyAttribute.cs /target:library /define:"DEBUG;TRACE" /nostdlib /reference:packages/FluentAssertions.3.5.0/lib/net45/FluentAssertions.dll /reference:packages/FluentAssertions.3.5.0/lib/net45/FluentAssertions.Core.dll /reference:packages/Gauge.CSharp.Lib.0.5.3.0/lib/net45/Gauge.CSharp.Lib.dll /reference:/usr/local/Cellar/mono/3.12.0/lib/mono/4.5/System.dll /reference:/usr/local/Cellar/mono/3.12.0/lib/mono/4.5/System.Xml.Linq.dll /reference:/usr/local/Cellar/mono/3.12.0/lib/mono/4.5/System.Data.DataSetExtensions.dll /reference:/usr/local/Cellar/mono/3.12.0/lib/mono/4.5/Microsoft.CSharp.dll /reference:/usr/local/Cellar/mono/3.12.0/lib/mono/4.5/System.Data.dll /reference:/usr/local/Cellar/mono/3.12.0/lib/mono/4.5/System.Xml.dll /reference:/usr/local/Cellar/mono/3.12.0/lib/mono/4.5/System.Core.dll /reference:/usr/local/Cellar/mono/3.12.0/lib/mono/4.5/mscorlib.dll /warn:4
Target DeployOutputFiles:
Copying file from '/private/tmp/bar/obj/Debug/bar.dll.mdb' to '/private/tmp/bar/bin/Debug/bar.dll.mdb'
Copying file from '/private/tmp/bar/obj/Debug/bar.dll' to '/private/tmp/bar/bin/Debug/bar.dll'
Done building project "/private/tmp/bar/bar.csproj".
Build succeeded.
0 Warning(s)
0 Error(s)
Time Elapsed 00:00:00.7010480
gauge specs
Output
INb:bar steam$ gauge_custom_build_path=bin/Debug gauge specs
Gauge.CSharp.Runner.SandboxFactory Creating a Sandbox in: /tmp/bar/bin/Debug
# Specification Heading
## Vowel counts in single word ✔ ✔
## Vowel counts in multiple word ✔ ✔
Successfully generated html-report to => /tmp/bar/reports/html-report
Specifications: 1 executed 1 passed 0 failed 0 skipped
Scenarios: 2 executed 2 passed 0 failed 0 skipped
Total time taken: 121ms
Hi @sriv,
Thank you for your help.
I have several projects which I maintain on linux and windows. At some point I started using F# FAKE and paket for build system and I'd say there is nothing to complain. So my suggestion on solving cross-platform build problems would be to leverage those projects.
GaugeProjectBuilder
I think it could be implemented using FAKEs msbuildhelper. After all this class responsibility is to find *.csproj
project file and build it.
Yep, everything in our build script needs to be made cross platform.
As I said, I think it is best to move to FAKE + paket. I can start that if you like.
Full paths
I checked again, it works. I must have done something wrong before.
I was able to execute gauge runner in IntegrationTestSample
and generate report. csharp.json
was
{
"id" : "csharp",
"version" : "0.7.2",
"description": "C# support for gauge",
"install": {
"windows": [],
"linux": [],
"darwin": []
},
"run": {
"windows": ["bin\\Gauge.Csharp.Runner.exe", "--start"],
"linux": ["/usr/bin/mono", "bin/Gauge.CSharp.Runner.exe", "--start"],
"darwin": ["echo", "Coming Soon!"]
},
"init": {
"windows": ["bin\\Gauge.Csharp.Runner.exe", "--init"],
"linux": ["echo", "Coming Soon!"],
"darwin": ["echo", "Coming Soon!"]
},
"gaugeVersionSupport": {
"minimum": "0.2.0",
"maximum": ""
}
}
FAKE + paket would be great, please proceed and keep us posted on how you get along with this. And thanks for signing up for this contribution, much appreciated.
I'll be working on convertion to FAKE and paket today.
@sriv can you tell me more about Gauge.CSharp.Core
project? Why is treated specially? Do you want it to be build together with all other projects? Does it need a nuget package?
@tomzo - Neat.
Regarding Gauge.CSharp.Core
- it contains some classes/methods that help communicate with Gauge Core's API. Two separate components use it: a) Gauge CSharp Runner and b)Gauge VisualStudio plugin. Hence it is extracted out as a separate project and hosted in Nuget just for ease of package management.
The core package is not supposed to be used by an end user, hence it is marked as "Internal".
Does this help?
Thanks. I am trying to understand what end results we want from build targets in this project.
So there should be
- one "chain" where I build
Gauge.Csharp.Core
project and package it as nuget. - one "chain" where I build
Gauge.Csharp.Lib
, run tests, and package it as another nuget. - finaly runner, which I build, unit test, integration test and package zip with the skeleton
Is that more or less correct?
That sounds like a good start, we need to have targets to run Functional Tests (refer here) and Regenerate proto classes (whenever there is any change in proto contracts), but I guess those can be added later.
Functional tests
OK. I don't think these will be a problem.
I have managed so far to create build and test tasks. Currently 26 tests are failing on linux due to path separators, I think these error are all because of asserts written with \
separator, so that will be easy to fix.
I am building this on linux and windows as I work. I got stuck about an hour ago with assembly loading problem on windows. Please have a look at output below. Did you have any errors like this before?
...
05:21:04.109 C:\GoAgent\pipelines\gauge-csharp\packages\test\NUnit.Runners\tools\nunit-console.exe "-nologo" "-noshadow" "-labels" "C:\GoAgent\pipelines\gauge-csharp\artifacts\gauge-csharp\bin\Gauge.CSharp.Runner.IntegrationTests.dll" "C:\GoAgent\pipelines\gauge-csharp\artifacts\gauge-csharp\bin\Gauge.CSharp.Runner.UnitTests.dll" "C:\GoAgent\pipelines\gauge-csharp\artifacts\gauge-csharp\bin\IntegrationTestSample.dll" "-xml:artifacts/gauge-csharp/bin/gauge.csharp.runner.unittests.xml"
05:21:04.406 ProcessModel: Default DomainUsage: Multiple
05:21:04.406 Execution Runtime: net-3.5
05:21:06.171 Unhandled Exception:
05:21:06.171 System.IO.FileLoadException: Could not load file or assembly 'Gauge.CSharp.Lib, Version=0.5.3.0, Culture=neutral, PublicKeyToken=3f92af01a0ea350e' or one of its dependencies. The located assembly's manifest definition does not match the assembly reference. (Exception from HRESULT: 0x80131040)
05:21:06.171 File name: 'Gauge.CSharp.Lib, Version=0.5.3.0, Culture=neutral, PublicKeyToken=3f92af01a0ea350e'
05:21:06.171
05:21:06.171 Server stack trace:
05:21:06.171 at System.ModuleHandle.ResolveType(RuntimeModule module, Int32 typeToken, IntPtr* typeInstArgs, Int32 typeInstCount, IntPtr* methodInstArgs, Int32 methodInstCount, ObjectHandleOnStack type)
05:21:06.171 at System.ModuleHandle.ResolveTypeHandleInternal(RuntimeModule module, Int32 typeToken, RuntimeTypeHandle[] typeInstantiationContext, RuntimeTypeHandle[] methodInstantiationContext)
05:21:06.171 at System.ModuleHandle.ResolveTypeHandle(Int32 typeToken, RuntimeTypeHandle[] typeInstantiationContext, RuntimeTypeHandle[] methodInstantiationContext)
05:21:06.171 at System.Reflection.RuntimeModule.ResolveType(Int32 metadataToken, Type[] genericTypeArguments, Type[] genericMethodArguments)
05:21:06.171 at System.Reflection.CustomAttribute.FilterCustomAttributeRecord(CustomAttributeRecord caRecord, MetadataImport scope, Assembly& lastAptcaOkAssembly, RuntimeModule decoratedModule, MetadataToken decoratedToken, RuntimeType attributeFilterType, Boolean mustBeInheritable, Object[] attributes, IList derivedAttributes, RuntimeType& attributeType, IRuntimeMethodInfo& ctor, Boolean& ctorHasParameters, Boolean& isVarArg)
05:21:06.171 at System.Reflection.CustomAttribute.GetCustomAttributes(RuntimeModule decoratedModule, Int32 decoratedMetadataToken, Int32 pcaCount, RuntimeType attributeFilterType, Boolean mustBeInheritable, IList derivedAttributes, Boolean isDecoratedTargetSecurityTransparent)
05:21:06.171 at System.Reflection.CustomAttribute.GetCustomAttributes(RuntimeMethodInfo method, RuntimeType caType, Boolean inherit)
05:21:06.171 at NUnit.Core.Reflect.GetAttributes(ICustomAttributeProvider member, Boolean inherit)
05:21:06.171 at NUnit.Core.Reflect.HasAttribute(ICustomAttributeProvider member, String attrName, Boolean inherit)
05:21:06.171 at NUnit.Core.Reflect.GetMethodsWithAttribute(Type fixtureType, String attributeName, Boolean inherit)
05:21:06.171 at NUnit.Core.NUnitTestFixture..ctor(Type fixtureType, Object[] arguments)
05:21:06.171 at NUnit.Core.Builders.NUnitTestFixtureBuilder.BuildSingleFixture(Type type, Attribute attr)
05:21:06.171 at NUnit.Core.Builders.NUnitTestFixtureBuilder.BuildFrom(Type type)
05:21:06.171 at NUnit.Core.Extensibility.SuiteBuilderCollection.BuildFrom(Type type)
05:21:06.171 at NUnit.Core.TestFixtureBuilder.BuildFrom(Type type)
05:21:06.171 at NUnit.Core.Builders.TestAssemblyBuilder.GetFixtures(Assembly assembly, String ns)
05:21:06.171 at NUnit.Core.Builders.TestAssemblyBuilder.Build(String assemblyName, Boolean autoSuites)
05:21:06.171 at NUnit.Core.Builders.TestAssemblyBuilder.Build(String assemblyName, String testName, Boolean autoSuites)
05:21:06.171 at NUnit.Core.TestSuiteBuilder.BuildSingleAssembly(TestPackage package)
05:21:06.171 at NUnit.Core.TestSuiteBuilder.Build(TestPackage package)
05:21:06.171 at NUnit.Core.SimpleTestRunner.Load(TestPackage package)
05:21:06.171 at NUnit.Core.ProxyTestRunner.Load(TestPackage package)
05:21:06.171 at NUnit.Core.ProxyTestRunner.Load(TestPackage package)
05:21:06.171 at NUnit.Core.RemoteTestRunner.Load(TestPackage package)
05:21:06.171 at System.Runtime.Remoting.Messaging.StackBuilderSink._PrivateProcessMessage(IntPtr md, Object[] args, Object server, Object[]& outArgs)
05:21:06.171 at System.Runtime.Remoting.Messaging.StackBuilderSink.SyncProcessMessage(IMessage msg)
05:21:06.171
05:21:06.171 Exception rethrown at [0]:
05:21:06.171 at System.Runtime.Remoting.Proxies.RealProxy.HandleReturnMessage(IMessage reqMsg, IMessage retMsg)
05:21:06.171 at System.Runtime.Remoting.Proxies.RealProxy.PrivateInvoke(MessageData& msgData, Int32 type)
05:21:06.171 at NUnit.Core.TestRunner.Load(TestPackage package)
05:21:06.171 at NUnit.Util.TestDomain.Load(TestPackage package)
05:21:06.171 at NUnit.Util.AggregatingTestRunner.Load(TestPackage package)
05:21:06.171 at NUnit.Core.ProxyTestRunner.Load(TestPackage package)
05:21:06.171 at NUnit.Util.RemoteTestAgent.AgentRunner.Load(TestPackage package)
05:21:06.171 at System.Runtime.Remoting.Messaging.StackBuilderSink._PrivateProcessMessage(IntPtr md, Object[] args, Object server, Object[]& outArgs)
05:21:06.171 at System.Runtime.Remoting.Messaging.StackBuilderSink.SyncProcessMessage(IMessage msg)
05:21:06.171
05:21:06.171 Exception rethrown at [1]:
05:21:06.171 at System.Runtime.Remoting.Proxies.RealProxy.HandleReturnMessage(IMessage reqMsg, IMessage retMsg)
05:21:06.171 at System.Runtime.Remoting.Proxies.RealProxy.PrivateInvoke(MessageData& msgData, Int32 type)
05:21:06.171 at NUnit.Core.TestRunner.Load(TestPackage package)
05:21:06.171 at NUnit.Core.ProxyTestRunner.Load(TestPackage package)
05:21:06.171 at NUnit.Util.ProcessRunner.Load(TestPackage package)
05:21:06.171 at NUnit.ConsoleRunner.ConsoleUi.Execute(ConsoleOptions options)
05:21:06.171 at NUnit.ConsoleRunner.Runner.Main(String[] args)
05:21:06.171
05:21:06.171 WRN: Assembly binding logging is turned OFF.
05:21:06.171 To enable assembly bind failure logging, set the registry value [HKLM\Software\Microsoft\Fusion!EnableLog] (DWORD) to 1.
05:21:06.171 Note: There is some performance penalty associated with assembly bind failure logging.
05:21:06.171 To turn this feature off, remove the registry value [HKLM\Software\Microsoft\Fusion!EnableLog].
05:21:06.171
05:21:06.234 Running build failed.
05:21:06.234 Error:
05:21:06.234 NUnit test failed. Process finished with exit code UnexpectedError (-100).
05:21:06.249
05:21:06.249 ---------------------------------------------------------------------
05:21:06.249 Build Time Report
05:21:06.249 ---------------------------------------------------------------------
05:21:06.249 No target was successfully completed
05:21:06.249 ---------------------------------------------------------------------
05:21:06.812 [script-executor] Script completed with exit code: 42.
This also happens when starting runner from command line, so I might be able to debug.
Another question, wouldn't it be easier if there was just one solution for all projects? Was there any particualr reason to have 5 solutions?
With 1 solution, all references could be by project. (not HintPath
)
So the problem is that, there is a commited binary at https://github.com/getgauge/gauge-csharp/blob/master/IntegrationTestSample/Lib/Gauge.CSharp.Lib.dll which was causing conflicts with newly built version of Gauge.CSharp.Lib.dll
.
The history of commits of that file does not say much, @sriv can you tell me what was the vision for lifecycle of this file? Would it be OK if I removed it from git and add a FAKE target that installs it from artifacts/
directory?
I also noticed, current distribution zip archives contain all test assemblies and even the xml output from test runs, do we really want them included? I can easily take of this.
Hi @tomzo , some responses:
wouldn't it be easier if there was just one solution for all projects? Was there any particualr reason to have 5 solutions
The reason for various solutions is to generate artifacts without having to compile other. ex. Gauge.CSharp.Core.sln
is used just to generate the Core nupkg.
I am happy to consolidate this, if you feel the newer build tasks could take care of atomic project packaging and generate the artifacts.
Committed
Gauge.CSharp.Lib.dll
.. Would it be OK if I removed it from git and add a FAKE target that installs it from artifacts/ directory?
So, the purpose of this checked in dll is just one - it serves as input for a test that ensures that Gauge.CSharp.Runner works when using a Lib
version that is different from the Lib
that the user's Gauge Csharp uses. If you notice the version of the checked-in Gauge.CSharp.Lib.dll
, it would be 0.0.0
. This test uses this information.
If you can think of an alternative to either mock/stub or simulate a different version of the dll, we could take this file out. Thoughts?
I also noticed, current distribution zip archives contain all test assemblies and even the xml output from test runs, do we really want them included? I can easily take of this.
Good catch, definitely not intentional. We should only package binaries required for execution.
Thanks for your answers, that clarifies a few things.
Gauge.CSharp.Lib.dll
So it works a little bit like regression test. It isn't so bad being in SCM after you explained it. However I cannot get that Runner.IntegrationTests
assembly to load on windows. The only time it worked, was when I replaced checked-in assembly with a new one, but I see now how that is wrong.
If you can think of an alternative to either mock/stub or simulate a different version of the dll, we could take this file out. Thoughts?
Fake has good support for generating AssemblyInfo with any version you want. So we could compile assembly twice with 2 different versions.
Another option would be to just add a strict reference in IntegrationTestSample on old already published nuget Gauge.CSharp.Lib package. E.g. we compile IntegrationTestSample
with Gauge.CSharp.Lib
0.0.2
and then test that such version was loaded. Do you think that would keep the value of tests?
I am going to leave the windows integration test problem for a moment.
I'll add zip and nuget packaging targets and make a PR with FAKE-only changes.
Then I will start working on fixing linux tests and approach the GaugeProjectBuilder
to make it mono-friendly.
BTW it seems, the result of these changes can be linux and Mac support, right?
However I cannot get that Runner.IntegrationTests assembly to load on windows
Strange, what do you see when you run .\build\test.ps
? It should run that test loading the checked-in Gauge.CSharp.Lib.dll
. This works on an appveyor build that we maintain parallel to our go.cd pipelines. [Note: the latest appveyor build is broken because of a change in the build scripts, will update now]
generating AssemblyInfo with any version you want v/s just add a strict reference in IntegrationTestSample on old already published package
I think the latter would be a better choice, there is no reason to build the dll fresh, with new versioninfo, this is more like reference data for the test hence, I would like it to be a static entity. What do you say?
I will start working on fixing linux tests and approach the GaugeProjectBuilder to make it mono-friendly.
Sounds good. Just some context, we initially were invoking msbuild.exe
and then moved to using the API via Microsoft.Build
namespace. It gave us more granular information about the build outcome and console logging.
the result of these changes can be linux and Mac support, right?
I believe so, at least it will be the bulk of what is required, we can iron out anything else that pops up.
On a side note, I just wanted to highlight that there is a CLA in place for Gauge, I hope you are OK to sign it?
Strange, what do you see when you run .\build\test.ps?
I don't run the powershell scripts, since I presume we want to move away from them. But that script does nothing more than runs nunit-console.exe with test assemblies paths, right?
Now there fake task, that does the same. You can try it out from my fake
branch. Just run
build.cmd CopyBinaries # here you should look what is in artifacts/ directory now
build.cmd RunTests # this will fail on integration tests in windows
I have already posted output from #74 (comment)
Quick google search says that these are issues with loading 2 version of the same assembly. Which makes some sense, the Sandbox
role is to load second application domain with older Lib, the one in gauge-bin
, right? But the problem here, seems to be that integration test is trying to load 0.0.0
version in the test runners domain.
I would like it to be a static entity. What do you say?
OK. I'd like to get tests working first, "the old way" before breaking more stuff anyway.
CLA
Sure. Signed.
New questions
Are you sure this is correct? Shouldn't it be
sandboxDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;
We want the sandbox domain to load assemblies from gauge bin directory. Not the main main domain where runner is executing. Or I misunderstood something...
Case sensitive environment variables
What is gauge statement about environment variables casing?
On linux there are failing tests because in test code there is
Environment.SetEnvironmentVariable("GAUGE_ADDITIONAL_LIBS", "foo.dll");
And then in runner project we are reading
Environment.GetEnvironmentVariable("gauge_additional_libs");
This may have worked on windows, but it won't on linux. My proposition is to make all environment variable reading case insensitive. That means replace all calls to GetEnvironmentVariable
by a method which wil first check "GAUGE_ADDITIONAL_LIBS"
, then "gauge_additional_libs"
. So that use can set either one of them to get desired result.
Another option is to just use upper case, always. I can do that too. Tell me what you think.
Thanks for the update, I am travelling for the rest of this week starting tomorrow evening, so may not be able to contribute much, apologies if this blocks your progress.
Now there fake task, that does the same. You can try it out from my fake branch.
I will try that out in a bit and respond.
resolving assemblies for SandboxDomain
I thought so too, but it didn't seem to work for me. I will have another look. Logically what you say makes sense, but I remember changing that caused the runner to crash. I won't be surprised if this is incorrect.
the Sandbox role is to load second application domain with older Lib, the one in gauge-bin, right?
Yes, the functional role of the Sandbox is to provide isolation between the test runner and user code's referenced assemblies. Ultimately, a test runner referencing Gauge.CSharp.Lib.dll version 0.5.1
should be able to execute test code referencing Gauge.CSharp.Lib.dll version 0.5.3
.
Also if you are changing/refactoring/thinking about the appdomains, please can you also look at #37? I am trying to find a clean way of honouring assembly binding-redirects.
Case sensitive environment variables
Agreed. Environment variables should be case sensitive and upper cased. I am happy to have gauge-csharp take care of case while reading the env vars. I think having a consistent convention that would work cross platform makes sense. Thanks for bringing this out.
Thanks for the update, I am travelling for the rest of this week starting tomorrow evening, so may not be able to contribute much, apologies if this blocks your progress.
I should be fine as long as you take a look from time to time and respond to some questions.
I thought so too, but it didn't seem to work for me. I will have another look. Logically what you say makes sense, but I remember changing that caused the runner to crash. I won't be surprised if this is incorrect.
IMHO currently the sandboxing is broken and never worked. I would speculate that only assembly that was loaded during integration tests so far was the one with 0.0.0
version, it was loaded in current and sandboxed domain. Assert is that there 0.0.0
loaded in sandbox, which was true. But there is check that test run has non 0.0.0
version. I think we can verify this on master by adding assert.
Also if you are changing/refactoring/thinking about the appdomains, please can you also look at #37? I am trying to find a clean way of honouring assembly binding-redirects.
I might have to, the SandBoxTests
are passing when I execute single case, but some are failing when I run them all at once. This happens on linux.
I finally have progress on loading assembly problems. Above clue was correct, basically the main domain (runner's domain) is at some point is loading (or at least trying to load) the users assembly. In tests that is IntegrationTestSample
and Gauge.CSharp.Lib version 0.0.0
.
And just to clarify, in .NET it is allowed to have 2 or more versions of the same assembly loaded into the same domain. But from my experience that will have horrible consequences for debugging and maintaince.
I got all tests working in mono first, I replaced that assembly resolution code, so that users methods are loaded only in the sandbox domain. Good, right?
But then I started working on windows and I apprently windows still hates me:
09:28:25.540 1) Test Error : Gauge.CSharp.Runner.IntegrationTests.SandBoxTests.ShouldGetAfterScenarioHooks
09:28:25.540 System.IO.FileLoadException : Could not load file or assembly 'Gauge.CSharp.Lib, Version=0.0.0.0, Culture=neutral, PublicKeyToken=3f92af01a0ea350e' or one of its dependencies. The located assembly's manifest definition does not match the assembly reference. (Exception from HRESULT: 0x80131040)
09:28:25.540
09:28:25.540 Server stack trace:
09:28:25.540 at System.Reflection.RuntimeAssembly._nLoad(AssemblyName fileName, String codeBase, Evidence assemblySecurity, RuntimeAssembly locationHint, StackCrawlMark& stackMark, IntPtr pPrivHostBinder, Boolean throwOnFileNotFound, Boolean forIntrospection, Boolean suppressSecurityChecks)
09:28:25.540 at System.Reflection.RuntimeAssembly.InternalLoadAssemblyName(AssemblyName assemblyRef, Evidence assemblySecurity, RuntimeAssembly reqAssembly, StackCrawlMark& stackMark, IntPtr pPrivHostBinder, Boolean throwOnFileNotFound, Boolean forIntrospection, Boolean suppressSecurityChecks)
09:28:25.540 at System.Reflection.RuntimeAssembly.InternalLoad(String assemblyString, Evidence assemblySecurity, StackCrawlMark& stackMark, IntPtr pPrivHostBinder, Boolean forIntrospection)
09:28:25.540 at System.Reflection.RuntimeAssembly.InternalLoad(String assemblyString, Evidence assemblySecurity, StackCrawlMark& stackMark, Boolean forIntrospection)
09:28:25.540 at System.Reflection.Assembly.Load(String assemblyString)
09:28:25.540 at System.UnitySerializationHolder.GetRealObject(StreamingContext context)
09:28:25.540 at System.Runtime.Serialization.ObjectManager.ResolveObjectReference(ObjectHolder holder)
09:28:25.540 at System.Runtime.Serialization.ObjectManager.DoFixups()
09:28:25.540 at System.Runtime.Serialization.Formatters.Binary.ObjectReader.Deserialize(HeaderHandler handler, __BinaryParser serParser, Boolean fCheck, Boolean isCrossAppDomain, IMethodCallMessage methodCallMessage)
09:28:25.540 at System.Runtime.Serialization.Formatters.Binary.BinaryFormatter.Deserialize(Stream serializationStream, HeaderHandler handler, Boolean fCheck, Boolean isCrossAppDomain, IMethodCallMessage methodCallMessage)
09:28:25.540 at System.Runtime.Remoting.Channels.CrossAppDomainSerializer.DeserializeObject(MemoryStream stm)
09:28:25.540 at System.Runtime.Remoting.Messaging.SmuggledMethodReturnMessage.FixupForNewAppDomain()
09:28:25.540 at System.Runtime.Remoting.Channels.CrossAppDomainSink.SyncProcessMessage(IMessage reqMsg)
09:28:25.540
09:28:25.540 Exception rethrown at [0]:
09:28:25.540 at System.Runtime.Remoting.Proxies.RealProxy.HandleReturnMessage(IMessage reqMsg, IMessage retMsg)
09:28:25.540 at System.Runtime.Remoting.Proxies.RealProxy.PrivateInvoke(MessageData& msgData, Int32 type)
09:28:25.540 at Gauge.CSharp.Runner.ISandbox.GetHookRegistry()
09:28:25.540 at Gauge.CSharp.Runner.IntegrationTests.SandBoxTests.ShouldGetAfterScenarioHooks() in c:\GoAgent\pipelines\gauge-csharp\Runner.IntegrationTests\SandBoxTests.cs:line 146
09:28:25.540
The important part is that it throws on Gauge.CSharp.Runner.ISandbox.GetHookRegistry()
. Why? Because there are elements of System.Reflection
in the ISandbox
which is used as contract in communication between the 2 domains.
public interface ISandbox
{
ExecutionResult ExecuteMethod(MethodInfo method, params object[] args);
bool TryScreenCapture(out byte[] screenShotBytes);
IHookRegistry GetHookRegistry();
List<MethodInfo> GetStepMethods();
void InitializeDataStore(string dataStoreType);
IEnumerable<string> GetStepTexts(MethodInfo stepMethod);
List<string> GetAllStepTexts();
Assembly TargetLibAssembly { get; }
void ClearObjectCache();
IEnumerable<string> GetAllPendingMessages();
}
In short, methods like List<string> GetAllStepTexts();
are OK because types are all primitives or contain only serializable system types. But method like List<MethodInfo> GetStepMethods();
contains ultra-deep object MethodInfo
which contains Assembly
instances, which eventually has instance Gauge.CSharp.Lib version 0.0.0
. So when runner calls Gauge.CSharp.Runner.ISandbox.GetHookRegistry()
on the remoting proxy, and Gauge.CSharp.Lib version 0.0.0
is returned then runner's side tries to load that assembly (only on windows), which fails because this assembly is in gauge-bin
and I am executing tests from artifacts/gauge-csharp/itest
where there is only Gauge.CSharp.Lib version 0.5.3
. So assembly loading fails with
The located assembly's manifest definition does not match the assembly reference
because it has loaded 0.5.3
version instead of 0.0.0
which was returned by ISandbox
instance.
A quick solution is to just let the users assemblies be loaded in runners, which is what I did here. So now when proxy returns something forcing Gauge.CSharp.Lib version 0.0.0
to load, then this method will find the 0.0.0
assembly from gauge-bin
directory.
In the end, tests are passing on linux and windows now. Hurray!
But I don't think this is correct solution. I think we need to change ISandbox contract into something else. I don't think it should return MethodInfo
elements, intead we should ask it to execute some of the runners routines. I don't know much about runner's or gauge's logic so I won't make more suggestions here. @sriv please think about it.
For those who want try out running gauge C# on linux or mac, you can try out from my branch.
To build and install from source:
git clone --branch mono-pr https://github.com/tomzo/gauge-csharp.git
cd gauge-csharp/
./build.sh Zip
gauge --install csharp -f artifacts/gauge-csharp/gauge-csharp-*.zip
Quick try out
mkdir -p /tmp/gauge-demo &&\
cd /tmp/gauge-demo &&\
gauge --init csharp &&\
gauge specs/
You should see
create manifest.json
create specs
create specs/example.spec
create env
create env/default
create env/default/default.properties
create gauge-demo
create Properties/AssemblyInfo.cs
create env/default/csharp.properties
create StepImplementation.cs
create packages.config
create GaugeDemo.csproj
create GaugeDemo.sln
Installing Nuget Package : FluentAssertions, version: 3.5.0
Installing Nuget Package : Gauge.CSharp.Lib, version: 0.5.3
Done Installing Nuget Package!
Successfully initialized the project. Run specifications with "gauge specs/"
Building project: /tmp/gauge-demo/GaugeDemo.csproj
xbuild /tmp/gauge-demo/GaugeDemo.csproj /t:Build /v:q /p:RestorePackages="False" /p:Configuration="Release" /p:Platform="Any CPU" /p:OutputPath="/tmp/gauge-demo/gauge-bin" /logger:Fake.MsBuildLogger+ErrorLogger,"/home/ide/.gauge/plugins/csharp/0.8.0/bin/FakeLib.dll"
xbuild /tmp/gauge-demo/GaugeDemo.csproj /t:Build /v:q /p:RestorePackages="False" /p:Configuration="Release" /p:Platform="Any CPU" /p:OutputPath="/tmp/gauge-demo/gauge-bin" /logger:Fake.MsBuildLogger+ErrorLogger,"/home/ide/.gauge/plugins/csharp/0.8.0/bin/FakeLib.dll"
XBuild Engine Version 12.0
Mono, Version 4.2.2.0
Copyright (C) 2005-2013 Various Mono authors
Gauge.CSharp.Runner.SandboxFactory Creating a Sandbox in: /home/ide/.gauge/plugins/csharp/0.8.0/bin/
# Specification Heading
## Vowel counts in single word ✔ ✔
## Vowel counts in multiple word ✔ ✔
Successfully generated html-report to => /tmp/gauge-demo/reports/html-report
Specifications: 1 executed 1 passed 0 failed 0 skipped
Scenarios: 2 executed 2 passed 0 failed 0 skipped
Total time taken: 1.122s
Now available via release 0.8.0