2015-11-16 11:21:57 -08:00
// Copyright (c) .NET Foundation and contributors. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
using System ;
2015-12-29 12:29:13 -08:00
using System.Collections.Generic ;
using System.Linq ;
2015-10-30 10:34:02 -07:00
using System.IO ;
2015-12-29 12:29:13 -08:00
using System.Text ;
2015-10-30 10:34:02 -07:00
using Microsoft.Dnx.Runtime.Common.CommandLine ;
using Microsoft.DotNet.Cli.Utils ;
2015-12-29 12:29:13 -08:00
using Microsoft.DotNet.ProjectModel ;
using NuGet.Frameworks ;
2015-10-30 10:34:02 -07:00
2015-11-06 16:11:09 -08:00
namespace Microsoft.DotNet.Tools.Repl.Csi
2015-10-30 10:34:02 -07:00
{
public sealed class Program
{
public static int Main ( string [ ] args )
{
DebugHelper . HandleDebugSwitch ( ref args ) ;
2015-12-29 15:46:45 -08:00
var app = new CommandLineApplication ( throwOnUnexpectedArg : false ) ;
2015-11-06 16:11:09 -08:00
app . Name = "dotnet repl csi" ;
app . FullName = "C# REPL" ;
app . Description = "C# REPL for the .NET platform" ;
2015-10-30 10:34:02 -07:00
app . HelpOption ( "-h|--help" ) ;
2015-12-29 12:29:13 -08:00
2015-12-30 09:41:45 -08:00
var script = app . Argument ( "<SCRIPT>" , "The .csx file to run. Defaults to interactive mode" ) ;
var framework = app . Option ( "-f|--framework <FRAMEWORK>" , "Compile a specific framework" , CommandOptionType . SingleValue ) ;
2015-12-29 12:29:13 -08:00
var configuration = app . Option ( "-c|--configuration <CONFIGURATION>" , "Configuration under which to build" , CommandOptionType . SingleValue ) ;
2015-12-30 09:41:45 -08:00
var preserveTemporary = app . Option ( "-t|--preserve-temporary" , "Preserve the temporary directory containing the compiled project" , CommandOptionType . NoValue ) ;
2015-12-29 12:29:13 -08:00
var project = app . Option ( "-p|--project <PROJECT>" , "The path to the project to run. Can be a path to a project.json or a project directory" , CommandOptionType . SingleValue ) ;
2015-10-30 10:34:02 -07:00
2015-12-30 09:41:45 -08:00
app . OnExecute ( ( ) = > Run ( script . Value , framework . Value ( ) , configuration . Value ( ) , preserveTemporary . HasValue ( ) , project . Value ( ) , app . RemainingArguments ) ) ;
2015-10-30 10:34:02 -07:00
return app . Execute ( args ) ;
}
2015-12-30 09:41:45 -08:00
private static ProjectContext GetProjectContext ( string targetFramework , string projectPath )
2015-12-29 12:29:13 -08:00
{
2015-12-30 09:41:45 -08:00
// Selecting the target framework for the project is done in the same manner as dotnet run. If a target framework
// was specified, we attempt to create a context for that framework (and error out if the framework is unsupported).
// Otherwise, we pick the first context supported by the project.
2015-12-29 12:29:13 -08:00
var contexts = ProjectContext . CreateContextForEachFramework ( projectPath ) ;
var context = contexts . First ( ) ;
2015-12-30 09:41:45 -08:00
if ( targetFramework ! = null )
2015-12-29 12:29:13 -08:00
{
2015-12-30 09:41:45 -08:00
var framework = NuGetFramework . Parse ( targetFramework ) ;
2015-12-29 12:29:13 -08:00
context = contexts . FirstOrDefault ( c = > c . TargetFramework . Equals ( framework ) ) ;
}
return context ;
}
private static CommandResult CompileProject ( ProjectContext projectContext , string configuration , out string tempOutputDir )
{
tempOutputDir = Path . Combine ( projectContext . ProjectDirectory , "bin" , ".dotnetrepl" , Guid . NewGuid ( ) . ToString ( "N" ) ) ;
Reporter . Output . WriteLine ( $"Compiling {projectContext.RootProject.Identity.Name.Yellow()} for {projectContext.TargetFramework.DotNetFrameworkName.Yellow()} to use with the {" C # REPL ".Yellow()} environment." ) ;
2015-12-30 09:41:45 -08:00
// --temp-output is actually the intermediate output folder and can be the same as --output for our temporary compilation (`dotnet run` can be seen doing the same)
return Command . Create ( $"dotnet-compile" , $"--output \" { tempOutputDir } \ " --temp-output \"{tempOutputDir}\" --framework \"{projectContext.TargetFramework}\" --configuration \"{configuration}\" \"{projectContext.ProjectDirectory}\"" )
2015-12-29 12:29:13 -08:00
. ForwardStdOut ( onlyIfVerbose : true )
. ForwardStdErr ( )
. Execute ( ) ;
}
2015-12-29 15:39:57 -08:00
private static IEnumerable < string > GetRuntimeDependencies ( ProjectContext projectContext , string buildConfiguration )
2015-12-29 12:29:13 -08:00
{
2015-12-30 09:41:45 -08:00
// We collect the full list of runtime dependencies here and pass them back so they can be
// referenced by the REPL environment when seeding the context. It appears that we need to
// explicitly list the dependencies as they may not exist in the output directory (as is the
// for library projects) or they may not exist anywhere on the path (e.g. they may only exist
// in the nuget package that was downloaded for the compilation) or they may be specific to a
// specific target framework.
2015-12-29 15:39:57 -08:00
var runtimeDependencies = new HashSet < string > ( ) ;
2015-12-29 12:29:13 -08:00
var projectExporter = projectContext . CreateExporter ( buildConfiguration ) ;
var projectDependencies = projectExporter . GetDependencies ( ) ;
foreach ( var projectDependency in projectDependencies )
{
var runtimeAssemblies = projectDependency . RuntimeAssemblies ;
foreach ( var runtimeAssembly in runtimeAssemblies )
{
var runtimeAssemblyPath = runtimeAssembly . ResolvedPath ;
2015-12-29 15:39:57 -08:00
runtimeDependencies . Add ( runtimeAssemblyPath ) ;
2015-12-29 12:29:13 -08:00
}
}
return runtimeDependencies ;
}
private static string CreateResponseFile ( ProjectContext projectContext , string buildConfiguration , string tempOutputDir )
{
var outputFileName = projectContext . ProjectFile . Name ;
var outputFilePath = Path . Combine ( tempOutputDir , $"{outputFileName}{Constants.DynamicLibSuffix}" ) ;
var projectResponseFilePath = Path . Combine ( tempOutputDir , $"dotnet-repl.{outputFileName}{Constants.ResponseFileSuffix}" ) ;
var runtimeDependencies = GetRuntimeDependencies ( projectContext , buildConfiguration ) ;
2015-12-30 09:41:45 -08:00
using ( var fileStream = new FileStream ( projectResponseFilePath , FileMode . Create ) )
2015-12-29 12:29:13 -08:00
{
2015-12-30 09:41:45 -08:00
using ( var streamWriter = new StreamWriter ( fileStream ) )
2015-12-29 12:29:13 -08:00
{
2015-12-30 09:41:45 -08:00
streamWriter . WriteLine ( $"/r:\" { outputFilePath } \ "" ) ;
foreach ( var projectDependency in runtimeDependencies )
{
streamWriter . WriteLine ( $"/r:\" { projectDependency } \ "" ) ;
}
2015-12-29 12:29:13 -08:00
}
}
return projectResponseFilePath ;
}
2015-12-30 09:41:45 -08:00
private static int Run ( string script , string targetFramework , string buildConfiguration , bool preserveTemporaryOutput , string projectPath , IEnumerable < string > remainingArguments )
2015-10-30 10:34:02 -07:00
{
var corerun = Path . Combine ( AppContext . BaseDirectory , Constants . HostExecutableName ) ;
var csiExe = Path . Combine ( AppContext . BaseDirectory , "csi.exe" ) ;
2015-12-29 12:29:13 -08:00
var csiArgs = new StringBuilder ( ) ;
2015-12-29 15:39:57 -08:00
if ( buildConfiguration = = null )
{
buildConfiguration = Constants . DefaultConfiguration ;
}
string tempOutputDir = null ;
2015-12-30 09:41:45 -08:00
try
2015-12-29 12:29:13 -08:00
{
2015-12-30 09:41:45 -08:00
if ( ! string . IsNullOrWhiteSpace ( projectPath ) )
2015-12-29 15:39:57 -08:00
{
2015-12-30 09:41:45 -08:00
var projectContext = GetProjectContext ( targetFramework , projectPath ) ;
2015-12-29 15:39:57 -08:00
2015-12-30 09:41:45 -08:00
if ( projectContext = = null )
{
Reporter . Error . WriteLine ( $"Unrecognized framework: {targetFramework.First()}" . Red ( ) ) ;
}
2015-12-29 12:29:13 -08:00
2015-12-30 09:41:45 -08:00
var compileResult = CompileProject ( projectContext , buildConfiguration , out tempOutputDir ) ;
2015-12-29 12:29:13 -08:00
2015-12-30 09:41:45 -08:00
if ( compileResult . ExitCode ! = 0 )
{
return compileResult . ExitCode ;
}
2015-12-29 12:29:13 -08:00
2015-12-30 09:41:45 -08:00
string responseFile = CreateResponseFile ( projectContext , buildConfiguration , tempOutputDir ) ;
csiArgs . Append ( $"@\" { responseFile } \ " " ) ;
}
2015-12-29 15:46:45 -08:00
2015-12-30 09:41:45 -08:00
if ( string . IsNullOrEmpty ( script ) & & ! remainingArguments . Any ( ) )
{
csiArgs . Append ( "-i" ) ;
}
else
{
csiArgs . Append ( script ) ;
}
2015-12-29 12:29:13 -08:00
2015-12-30 09:41:45 -08:00
foreach ( string remainingArgument in remainingArguments )
{
csiArgs . Append ( $" {remainingArgument}" ) ;
}
2015-12-29 12:29:13 -08:00
2015-12-30 09:41:45 -08:00
return Command . Create ( csiExe , csiArgs . ToString ( ) )
. ForwardStdOut ( )
. ForwardStdErr ( )
. Execute ( )
. ExitCode ;
}
finally
2015-12-29 12:29:13 -08:00
{
2015-12-30 09:41:45 -08:00
if ( ( tempOutputDir ! = null ) & & ! preserveTemporaryOutput )
{
Directory . Delete ( tempOutputDir , recursive : true ) ;
}
2015-12-29 12:29:13 -08:00
}
2015-10-30 10:34:02 -07:00
}
}
}