Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added support for referencing NuGet packages in C# scripts #813

Merged
merged 22 commits into from
Jun 15, 2017
Merged

Added support for referencing NuGet packages in C# scripts #813

merged 22 commits into from
Jun 15, 2017

Conversation

seesharper
Copy link
Contributor

@seesharper seesharper commented Apr 4, 2017

This PR adds support for resolving NuGet packages in C# scripts

Example

#r "nuget:AutoMapper/6.0.0"

The feature is implemented by decorating the default MetadataReferenceResolver with a resolver that will know how to handle references to NuGet packages. The code for the NuGetMetadataReferenceResolver is currently maintained at this repo

https://github.com/seesharper/Dotnet.Script.NuGetMetadataResolver

The reason for keeping the NuGetMetadataReferenceResolver outside of this project is so that it can be used by other script runners such as https://github.com/filipw/dotnet-script

@@ -79,16 +89,29 @@ public class ScriptProjectSystem : IProjectSystem
private readonly Dictionary<string, ProjectInfo> _projects;
private readonly OmniSharpWorkspace _workspace;
private readonly IOmniSharpEnvironment _env;
private readonly ILoggerFactory loggerFactory;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add an underscore to match the other fields in this file.

var resolver = NuGetMetadataReferenceResolver.Create(ScriptMetadataResolver.Default,
NugetFrameworkProvider.GetFrameworkNameFromAssembly(),loggerFactory, _env.Path);


Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

remove extra blank line.

@@ -18,6 +18,9 @@

namespace OmniSharp.Script
{
using Dotnet.Script.NuGetMetadataResolver;
using Microsoft.DotNet.InternalAbstractions;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please don't nest usings inside of namespaces. Put them with the rest of the namespaces at the top of the file.

private static Dictionary<string, ImmutableArray<PortableExecutableReference>> DirectReferenceCache = new Dictionary<string, ImmutableArray<PortableExecutableReference>>();
private static Dictionary<string, PortableExecutableReference> MissingReferenceCache = new Dictionary<string, PortableExecutableReference>();
private static MetadataReferenceResolver _defaultRuntimeResolver = ScriptMetadataResolver.Default;


Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

remove extra blank line

{
var resolver = NuGetMetadataReferenceResolver.Create(ScriptMetadataResolver.Default,
NugetFrameworkProvider.GetFrameworkNameFromAssembly(),loggerFactory, _env.Path);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

missing space between , and loggerFactory


public CachingScriptMetadataResolver(MetadataReferenceResolver defaultReferenceResolver)
{
this.defaultReferenceResolver = defaultReferenceResolver;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please use underscore for private field name.

metadataReferenceResolver:
new CachingScriptMetadataResolver(resolver),
sourceReferenceResolver: ScriptSourceResolver.Default,
assemblyIdentityComparer: DesktopAssemblyIdentityComparer.Default).
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you like, maybe move the . to the next line like you've done elsewhere. That always looks better to me.


foreach (var inheritedAssemblyName in inheritedAssemblyNames)
{
var assembly = Assembly.Load(inheritedAssemblyName);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We generally use an IAssemblyLoader so that we can track assemblies that are loaded at runtime. Just import IAssemblyLoader in the constructor and use it here.

{
var resolver = NuGetMetadataReferenceResolver.Create(ScriptMetadataResolver.Default,
NugetFrameworkProvider.GetFrameworkNameFromAssembly(),loggerFactory, _env.Path);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

_env.Path changed to _env.TargetDirectory in dev. That's why the build is failing.

@DustinCampbell DustinCampbell requested a review from filipw April 4, 2017 19:26
@DustinCampbell
Copy link
Contributor

Thanks for the PR! My feedback is mostly nits to try and match the code style of the files you're changing. Also, the CI build is failing because IOmniSharpEnvironment.Path changed to IOmniSharpEnvironment.TargetDirectory.

@seesharper
Copy link
Contributor Author

seesharper commented Apr 4, 2017

Thanks for your feedback. I think I got them issues sorted out now :)

@filipw
Copy link
Member

filipw commented Apr 4, 2017

@seesharper thanks a lot for the PR, always great to see people investing their time in CSX tooling 😀

I'm gonna loop in @tmat for his comments on this. Tomas, does this align with the Roslyn vision of supporting nuget references (seems to me it does)? In general we want OmniSharp to stay aligned with CSI.exe scripting. In this case, we can get ahead of the curve a little bit, which I think would be fine, as long as syntax-wise this is what you have thought of for Roslyn down the road?

I also like that it's a done as only a custom MetadataReferenceResolver rather than a ton of machinery for injecting stuff into ScriptOptions and things like that.


private static string ResolveTargetFramework()
{
return Assembly.GetEntryAssembly().GetCustomAttributes()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what problem did you run into that you added this code?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No problem. I was just making sure that I did not change the way metadata references are being added when running on the full desktop framework.

@@ -47,7 +51,7 @@ public override ImmutableArray<PortableExecutableReference> ResolveReference(str
return DirectReferenceCache[key];
}

var result = _defaultRuntimeResolver.ResolveReference(reference, baseFilePath, properties);
var result = _defaultReferenceResolver.ResolveReference(reference, baseFilePath, properties);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it's not visible what happens here for nuget packages, can you explain a little? do they get installed? what if there is no network? what if there is no nuget in the user's path? what's the lock file?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Any other feedback to @filipw's question above?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@seesharper: did this ever get answered?

{
var resolver = NuGetMetadataReferenceResolver.Create(ScriptMetadataResolver.Default,
NugetFrameworkProvider.GetFrameworkNameFromAssembly(), _loggerFactory, _env.TargetDirectory);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this where the nuget framework to be used is determined? if so, I don't think it's correct to assume the version of nuget framework from the version of OmniSharp. A desktop OmniSharp (in fact, in VS Code only that version is used) can be used to provide language services for a .NET Core script and vice versa, a .NET Core OmniSharp can be used to provide language services for a destkop framework script

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

How can we within OmniSharp determine if the we a running the script in the context of a .Net Core app when there is no project context? The whole idea here is to support "self-contained" scripts that has no additional context other than the script itself.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

so far - because CSI is desktop FX only and there was no formal specification for how .NET Core scripting should behave, OmniSharp would assume desktop FX mode for scripts in case no project.json was there, which would be the typical majority use case.

if project.json was found, then the first runtime FX defined there would be used.
in other words, the only way to force .NET Core scripting was through an external file, and I think for now this is how it should stay because it is non-standard scripting anyway.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I understand that this is a problem and that is why .Net Core metadata references are only added when OmniSharp itself is running as a .Net Core app. It's not ideal, but at least it would provide a way for people wanting to try out project-less .Net Core scripting by pointing to the .Net Core OmniSharp binary from VsCode. Also commented on an alternative approach here .

#813 (comment)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

C# for VS Code does not ship a .NET Core version of OmniSharp. It runs OmniSharp with an embedded Mono runtime. So, this approach will not work unless the user downloads or builds a .NET Core-based OmniSharp and sets VS Code to use that. And, if they do that, they may have other problems since .NET Core OmniSharp can't handle csproj projects properly.

Note, this will also be a problem long term as the .NET Core versions of OmniSharp will be going away in favor of a Mono-based approach. See #666 for explanation.

Copy link
Contributor Author

@seesharper seesharper Apr 5, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I sort of understood #666 wouldn't help me much :)
Jokes aside, is there anything we can do to hint OmniSharp about the scripting runtime?

What if we did this

#r "runtime:NetCoreApp1.1"

In the case of a non-existing project.json, we could create one at runtime based on the runtime reference and feed that into the existing logic that resolves metadata references based on project context,

In other words we create a project context based on script runtime references.

The really nice thing about this approach is that we could also add the NuGet references found in the script to the dynamically created project.json and basically skip the NuGetMetadataReferenceResolver all together.

@filipw We could do the exact same approach in Dotnet-Script with NuGet references. Only difference here is that we actually know the scripting runtime context.

@filipw
Copy link
Member

filipw commented Apr 4, 2017

Also, let's wait for input from @tmat but I think it would be better to delimit version with a comma. This is the design used in #r implementation in F# https://github.com/fsharp/fslang-design/blob/master/RFCs/FS-1027-fsi-references.md#summary

@seesharper
Copy link
Contributor Author

Yes, that was the general idea:) I will get back to you tomorrow with more details around how I ended up with this approach. 11pm here now :)

@tmat
Copy link
Contributor

tmat commented Apr 4, 2017

        if (DirectReferenceCache.ContainsKey(key))

It would be better to use a tuple here.


Refers to: src/OmniSharp.Script/CachingScriptMetadataResolver.cs:49 in f1d6dd7. [](commit_id = f1d6dd7, deletion_comment = False)

@tmat
Copy link
Contributor

tmat commented Apr 4, 2017

There is a related discussion in F# design repo: fsharp/fslang-design#167

Currently the syntax F# is considering is:

#r "nuget: <package-name>, <package-version>"

It would be good if C# and F# used the same syntax.

To make this work for all nupkgs (specifically with CoreFX nupkgs) we'll need to be able to deal with both reference assemblies and implementation assemblies in the resolver. Also someone pointed out that batching the resolution (i.e. have an API that allows the compiler to resolve multiple #r's in one go) would be necessary to avoid perf issues with discovering dependencies.

@tmat
Copy link
Contributor

tmat commented Apr 4, 2017

It would be useful to write up some specification, how the resolver works. For example, does it check NuGet.config files for feeds, how is the lookup cached across script executions -- is there a script.assets.json file that csi /restore script.csx would create, etc.?

@seesharper
Copy link
Contributor Author

seesharper commented Apr 5, 2017

Syntax

I did not have much to go after with regards to the syntax other than this issue

Add NuGet support in scripting APIs #6900

Simply adopted the suggested syntax with the assumption that this might be implemented in a future release of Roslyn.

I do however agree that we should align with F# and adopt the same syntax

#r "nuget:AutoMapper,6.0.0"

Resolver - How does it work

The NuGetMetadataReferenceResolver looks for #r references that starts with the word "nuget" and then parses the package name and version from that reference.

It then looks for the package in the global nuget cache and from here two things can happen.

Either the package is found in the cache and we reference the package from the global cache.

If the package is not found in the global cache, we install the package into a temporary folder which gets deleted after installing. This causes the package also to be installed into the global cache and we can again reference the package from the global cache.

The process of actually installing the package is offloaded to the NuGet command line application as NuGet does not offer a .Net Core version of NuGet.PackageManagement

After the package is resolved from the global cache we will look for the nearest matching framework from that package. This is done using the FrameworkReducer class provided by the NuGet API.

The target framework, meaning the context for which to find the nearest matching framework is either inferred from the host process or it can be specified when creating the resolver.

It is important to notice that the resolver does not make any attempts to resolve the package dependencies. Since there is no lock file in play here, it made more sense to explicitly (#r)eference all dependencies including transients.

It that sense you could say that the script is also the lock file.

The resolver uses the PackageSourceProvider (also provided by the NuGet API) to figure out the feeds to be used. This for instance means that we support a local NuGet.Config.

@seesharper
Copy link
Contributor Author

The challenge , revisited

  • Being able provide language services for scripts even if they provide no context in the form of a project.json/csproj file.

  • Being able to resolve references to NuGet packages in C# scripts.

So how do OmniSharp know if it should provide metadata for the full framework or .Net Core?

The PR uses information from the host process (OmniSharp) to determine if we are running .Net Core or the full framework. This is probably not good enough as @filipw pointed out as OmniSharp can provide language services for both runtimes regardless of the runtime that OmniSharp itself runs in.

So how can we provide OmniSharp with enough information about script target runtime?

What if we did something similar to SheBang on nix systems?

#!/bin/sh

we could do

#r "runtime:NetCoreApp1.1"

In my head, it sort of makes sense that we "reference" the runtime.

If we find multiple csx files with different "runtimes" we would have to handle this somehow though.

@filipw
Copy link
Member

filipw commented Apr 5, 2017

So how do OmniSharp know if it should provide metadata for the full framework or .Net Core?

I think for the sake of baby steps and incremental progress we could consider for now that for no known runtime context (such as CSI scripts) we just assume latest desktop FX, but maybe I'm being overly simplistic.
The natural next step would then be to graduate the discussion to how the CSX could manifest it's runtime environment.

I fear that if we dive into this now we will never get anything out as the scope would just be too large; not to mention such discussion really belongs in the Roslyn repo as that's the codebase that will drive this, and OmniSharp would merely consume it.

@tmat
Copy link
Contributor

tmat commented Apr 5, 2017

@seesharper Great write up. Could you add it to the readme.md of https://github.com/seesharper/Dotnet.Script.NuGetMetadataResolver?

Re the resolver implementation: One thing I'm worried about is specifying all transitive dependencies explicitly. That's not very script-friendly. Note that #r "assembly.dll" currently automatically references all transitive dependencies. So it would be odd if #r "nuget" didn't do that. Also inconvenient for the user, especially in the REPL. I'd prefer if the resolver found all dependencies and stored them in a .json file next to the script (this would be overridable API of the resolver, so that the host can do it in-memory, or write it to some host specific location). If such file already exists it would just use it and skip the dependency resolution unless #r nuget directives in the script changed.

Re runtime: We already have support for #! -- it's just a comment for the compiler and thus ignored. But I imagine we can come up with some convention for specifying the host environment. I don't think we should be specifying runtime (Core CLR vs Desktop CLR) though but rather the target API set. For this purpose using #r with a NETStandard.Library" meta-package seems most fitting.

For example,
#r "nuget:NETStandard.Library, 1.6.1"

I think there are two categories of scripts:

  1. csi runnable scripts
    These are universal scripts that might use host variables and default imports defined by csi (such as Args) but require no other host-specific types/variables/imports.

  2. custom host scripts
    These are host-specific scripts that assume an environment created by some application that runs them. This is where we could use #!. Perhaps we could develop some convention that allows the app author to specify what the script environment looks like statically, to enable editors to provide intellisense.

@seesharper
Copy link
Contributor Author

Before discussing further I would just like to show you what kind of script developer happiness I'm aiming
for here. No project file or any other files besides the actual script file.

nuget

seesharper added a commit to seesharper/Dotnet.Script.NuGetMetadataResolver that referenced this pull request Apr 7, 2017
@seesharper
Copy link
Contributor Author

I am not giving up on this yet :)
I have created a new version of DotNet.Script.NuGetMetadataResolver that takes a new approach to resolving NuGet packages. Updated readme at https://github.com/seesharper/Dotnet.Script.NuGetMetadataResolver

Question is, should I abandon this PR and start a new PR based on the latest sources?

@seesharper
Copy link
Contributor Author

seesharper commented Apr 24, 2017

@tmat @DustinCampbell @filipw Could you take another look at this in the light of the changes I've made to
https://github.com/seesharper/Dotnet.Script.NuGetMetadataResolver and the simplified ScriptProjectSystem?

@seesharper
Copy link
Contributor Author

@tmat @DustinCampbell @filipw What do you say, guys. Is this happening or not?

Copy link
Member

@david-driscoll david-driscoll left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm fine with the changes in essence, will wait for more feedback from @DustinCampbell or @filipw however.

NuGet.Config Outdated
</packageSources>
<disabledPackageSources>
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These probably shouldn't be committed (looks like custom sources you have locally)

@@ -47,7 +51,7 @@ public override ImmutableArray<PortableExecutableReference> ResolveReference(str
return DirectReferenceCache[key];
}

var result = _defaultRuntimeResolver.ResolveReference(reference, baseFilePath, properties);
var result = _defaultReferenceResolver.ResolveReference(reference, baseFilePath, properties);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Any other feedback to @filipw's question above?

}

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Minor nit, there is extra whitespace here, that isn't needed.

@seesharper
Copy link
Contributor Author

FYI: Just tested this on a Mac and it works equally well there.

@filipw
Copy link
Member

filipw commented Jun 7, 2017

@seesharper thanks - looks got to me. There are still some small things @DustinCampbell pointed out but after that, as far as I'm concerned, we could take this in as an experimental scripting feature 👍

@DustinCampbell @david-driscoll what do you guys think?

@filipw
Copy link
Member

filipw commented Jun 7, 2017

actually one more thing - looks like the configuration is not being consumed at the moment. Did you forget a push?

@seesharper
Copy link
Contributor Author

@filipw @DustinCampbell @david-driscoll

The configuration is being consumed here

https://github.com/OmniSharp/omnisharp-roslyn/pull/813/files#diff-ed5a3c116a19423c18b1e630f80455b9R77

I have also updated to the latest version of Dotnet.Script.NuGetMetadataResolver (2.0.3) and this PR should be able to be merged.

@filipw
Copy link
Member

filipw commented Jun 7, 2017

I meant that NuGetMetadataReferenceResolver shouldn't be used at all if the setting is false

@seesharper
Copy link
Contributor Author

Ah, I see. The ScriptHelper class is a static class. Is there any way that I can access the configuration from within the ScriptHelper class?

@seesharper
Copy link
Contributor Author

@filipw Probably need to make the ScriptHelper class an instance class in order to pass in "IConfiguration" ?

@filipw
Copy link
Member

filipw commented Jun 7, 2017

that's OK - it could be a singleton similar to i.e. https://github.com/OmniSharp/omnisharp-roslyn/blob/dev/src/OmniSharp.Roslyn/MetadataHelper.cs#L32 +

var metadataHelper = new MetadataHelper(loader);

@seesharper
Copy link
Contributor Author

@filipw Now the NuGetMetadataResolver is only used when the "enableScriptNuGetReferences" option is "true"

@seesharper
Copy link
Contributor Author

@filipw @DustinCampbell @david-driscoll
Would appreciate you taking the time to review the (hopefully) final changes.

@seesharper
Copy link
Contributor Author

@filipw @DustinCampbell @david-driscoll
Is this happening, guys? It's getting a little tedious to constantly have to keep this branch up-to-date.

Copy link
Contributor

@DustinCampbell DustinCampbell left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm comfortable with this change so long as @filipw is good with it.

@seesharper
Copy link
Contributor Author

Thanks @DustinCampbell . Build passing. @david-driscoll , @filipw . Are we merging this now?

Copy link
Member

@filipw filipw left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM

@filipw
Copy link
Member

filipw commented Jun 13, 2017

The merge button is unavailable until @david-driscoll approves

@seesharper
Copy link
Contributor Author

@david-driscoll Would really appreciate your final approval on this PR

@david-driscoll
Copy link
Member

Sorry missed the final approvals here... LGTM

@david-driscoll david-driscoll merged commit 8d9191b into OmniSharp:dev Jun 15, 2017
@seesharper
Copy link
Contributor Author

Thanks a lot, @david-driscoll. Appreciate it 👍

@adamralph
Copy link
Contributor

This is great! Is there an issue anywhere tracking the addition of this to csi.exe?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

7 participants