using System; using System.Collections.Generic; using System.Globalization; using System.IO; using System.Linq; using System.Web.Hosting; using Orchard.CodeGeneration.Services; using Orchard.Commands; using Orchard.Data.Migration.Generator; using Orchard.Data.Migration.Schema; using Orchard.Environment.Extensions; using Orchard.Environment.Extensions.Models; namespace Orchard.CodeGeneration.Commands { public class CodeGenerationCommands : DefaultOrchardCommandHandler { private readonly IExtensionManager _extensionManager; private readonly ISchemaCommandGenerator _schemaCommandGenerator; private const string SolutionDirectoryModules = "E9C9F120-07BA-4DFB-B9C3-3AFB9D44C9D5"; private const string SolutionDirectoryTests = "74E681ED-FECC-4034-B9BD-01B0BB1BDECA"; private const string SolutionDirectoryThemes = "74492CBC-7201-417E-BC29-28B4C25A58B0"; private static readonly string[] _themeDirectories = new[] { "", "Assets", "Content", "Styles", "Scripts", "Views" }; private static readonly string[] _moduleDirectories = new[] { "", "Properties", "Assets", "Controllers", "Views", "Models", "Scripts", "Styles" }; private static readonly string[] _moduleTestsDirectories = new[] { "", "Properties" }; private const string ModuleName = "CodeGeneration"; private static readonly string _codeGenTemplatePath = HostingEnvironment.MapPath("~/Modules/Orchard." + ModuleName + "/CodeGenerationTemplates/"); private static readonly string _orchardWebProj = HostingEnvironment.MapPath("~/Orchard.Web.csproj"); private static readonly string _orchardThemesProj = HostingEnvironment.MapPath("~/Themes/Themes.csproj"); public CodeGenerationCommands( IExtensionManager extensionManager, ISchemaCommandGenerator schemaCommandGenerator) { _extensionManager = extensionManager; _schemaCommandGenerator = schemaCommandGenerator; // Default is to include in the solution when generating modules / themes IncludeInSolution = true; } [OrchardSwitch] public bool IncludeInSolution { get; set; } [OrchardSwitch] public bool CreateProject { get; set; } [OrchardSwitch] public string BasedOn { get; set; } [CommandHelp("codegen datamigration \r\n\t" + "Create a new Data Migration class")] [CommandName("codegen datamigration")] public void CreateDataMigration(string featureName) { Context.Output.WriteLine(T("Creating Data Migration for {0}", featureName)); ExtensionDescriptor extensionDescriptor = _extensionManager.AvailableExtensions().FirstOrDefault(extension => DefaultExtensionTypes.IsModule(extension.ExtensionType) && extension.Features.Any(feature => String.Equals(feature.Id, featureName, StringComparison.OrdinalIgnoreCase))); if (extensionDescriptor == null) { Context.Output.WriteLine(T("Creating data migration failed: target Feature {0} could not be found.", featureName)); return; } string dataMigrationFolderPath = HostingEnvironment.MapPath("~/Modules/" + extensionDescriptor.Id + "/"); string dataMigrationFilePath = dataMigrationFolderPath + "Migrations.cs"; string templatesPath = HostingEnvironment.MapPath("~/Modules/Orchard." + ModuleName + "/CodeGenerationTemplates/"); string moduleCsProjPath = HostingEnvironment.MapPath(string.Format("~/Modules/{0}/{0}.csproj", extensionDescriptor.Id)); if (!Directory.Exists(dataMigrationFolderPath)) { Directory.CreateDirectory(dataMigrationFolderPath); } if (File.Exists(dataMigrationFilePath)) { Context.Output.WriteLine(T("Data migration already exists in target Module {0}.", extensionDescriptor.Id)); return; } List commands = _schemaCommandGenerator.GetCreateFeatureCommands(featureName, false).ToList(); string dataMigrationText; using (var stringWriter = new StringWriter()) { var interpreter = new CodeGenerationCommandInterpreter(stringWriter); foreach (var command in commands) { interpreter.Visit(command); stringWriter.WriteLine(); } dataMigrationText = File.ReadAllText(templatesPath + "DataMigration.txt"); dataMigrationText = dataMigrationText.Replace("$$FeatureName$$", featureName); dataMigrationText = dataMigrationText.Replace("$$Commands$$", stringWriter.ToString()); } File.WriteAllText(dataMigrationFilePath, dataMigrationText); string projectFileText = File.ReadAllText(moduleCsProjPath); // The string searches in solution/project files can be made aware of comment lines. if (projectFileText.Contains("\r\n ", "Migrations.cs"); projectFileText = projectFileText.Insert(projectFileText.LastIndexOf("\r\n \r\n \r\n ", "Migrations.cs"); projectFileText = projectFileText.Insert(projectFileText.LastIndexOf(""), itemGroupReference); } File.WriteAllText(moduleCsProjPath, projectFileText); TouchSolution(Context.Output); Context.Output.WriteLine(T("Data migration created successfully in Module {0}", extensionDescriptor.Id)); } [CommandHelp("codegen module [/IncludeInSolution:true|false]\r\n\t" + "Create a new Orchard module")] [CommandName("codegen module")] [OrchardSwitches("IncludeInSolution")] public void CreateModule(string moduleName) { Context.Output.WriteLine(T("Creating Module {0}", moduleName)); if (_extensionManager.AvailableExtensions().Any(extension => String.Equals(moduleName, extension.Name, StringComparison.OrdinalIgnoreCase))) { Context.Output.WriteLine(T("Creating Module {0} failed: a module of the same name already exists", moduleName)); return; } IntegrateModule(moduleName); Context.Output.WriteLine(T("Module {0} created successfully", moduleName)); } [CommandHelp("codegen moduletests [/IncludeInSolution:true|false]\r\n\t" + "Creates a new test project for a module")] [CommandName("codegen moduletests")] [OrchardSwitches("IncludeInSolution")] public void CreateModuleTests(string moduleName) { var projectName = moduleName + ".Tests"; Context.Output.WriteLine(T("Creating module tests project {0}", projectName)); var testsPath = HostingEnvironment.MapPath("~/Modules/" + moduleName + "/" + projectName + "/"); if (Directory.Exists(testsPath)) { Context.Output.WriteLine(T("Creating module tests project {0} failed: a project of the same name already exists", projectName)); return; } var propertiesPath = testsPath + "Properties"; var content = new HashSet(); var folders = new HashSet(); foreach (var folder in _moduleTestsDirectories) { Directory.CreateDirectory(testsPath + folder); if (!String.IsNullOrEmpty(folder)) { folders.Add(testsPath + folder); } } var projectGuid = Guid.NewGuid().ToString().ToUpper(); var templateText = File.ReadAllText(_codeGenTemplatePath + "ModuleAssemblyInfo.txt"); templateText = templateText.Replace("$$ModuleName$$", projectName); templateText = templateText.Replace("$$ModuleTypeLibGuid$$", Guid.NewGuid().ToString()); File.WriteAllText(propertiesPath + "\\AssemblyInfo.cs", templateText); content.Add(propertiesPath + "\\AssemblyInfo.cs"); File.WriteAllText(testsPath + "packages.config", File.ReadAllText(_codeGenTemplatePath + "ModuleTestsPackagesConfig.txt")); content.Add(testsPath + "packages.config"); var itemGroup = CreateProjectItemGroup(testsPath, content, folders); var csprojText = File.ReadAllText(_codeGenTemplatePath + "\\ModuleTestsCsProj.txt"); csprojText = csprojText.Replace("$$ProjectName$$", projectName); csprojText = csprojText.Replace("$$TestsProjectGuid$$", projectGuid); csprojText = csprojText.Replace("$$FileIncludes$$", itemGroup ?? ""); csprojText = csprojText.Replace("$$OrchardReferences$$", GetOrchardReferences(modulesFolderRelativeDepth: 3)); File.WriteAllText(testsPath + projectName + ".csproj", csprojText); // The string searches in solution/project files can be made aware of comment lines. if (IncludeInSolution) { AddToSolution(Context.Output, projectName, projectGuid, "Modules\\" + moduleName, SolutionDirectoryTests); } Context.Output.WriteLine(T("Module tests project {0} created successfully", projectName)); } [CommandName("codegen theme")] [CommandHelp("codegen theme [/CreateProject:true|false][/IncludeInSolution:true|false][/BasedOn:]\r\n\tCreate a new Orchard theme")] [OrchardSwitches("IncludeInSolution,BasedOn,CreateProject")] public void CreateTheme(string themeName) { Context.Output.WriteLine(T("Creating Theme {0}", themeName)); if (_extensionManager.AvailableExtensions().Any(extension => String.Equals(themeName, extension.Id, StringComparison.OrdinalIgnoreCase))) { Context.Output.WriteLine(T("Creating Theme {0} failed: an extention of the same name already exists", themeName)); return; } if (!string.IsNullOrEmpty(BasedOn)) { if (!_extensionManager.AvailableExtensions().Any(extension => string.Equals(extension.ExtensionType, DefaultExtensionTypes.Theme, StringComparison.OrdinalIgnoreCase) && string.Equals(BasedOn, extension.Id, StringComparison.OrdinalIgnoreCase))) { Context.Output.WriteLine(T("Creating Theme {0} failed: base theme named {1} was not found.", themeName, BasedOn)); return; } } IntegrateTheme(themeName, BasedOn); Context.Output.WriteLine(T("Theme {0} created successfully", themeName)); } [CommandHelp("codegen controller \r\n\t" + "Create a new Orchard controller in a module")] [CommandName("codegen controller")] public void CreateController(string moduleName, string controllerName) { Context.Output.WriteLine(T("Creating Controller {0} in Module {1}", controllerName, moduleName)); ExtensionDescriptor extensionDescriptor = _extensionManager.AvailableExtensions().FirstOrDefault(extension => DefaultExtensionTypes.IsModule(extension.ExtensionType) && string.Equals(moduleName, extension.Name, StringComparison.OrdinalIgnoreCase)); if (extensionDescriptor == null) { Context.Output.WriteLine(T("Creating Controller {0} failed: target Module {1} could not be found.", controllerName, moduleName)); return; } string moduleControllersPath = HostingEnvironment.MapPath("~/Modules/" + extensionDescriptor.Id + "/Controllers/"); string controllerPath = moduleControllersPath + controllerName + ".cs"; string moduleCsProjPath = HostingEnvironment.MapPath(string.Format("~/Modules/{0}/{0}.csproj", extensionDescriptor.Id)); string templatesPath = HostingEnvironment.MapPath("~/Modules/Orchard." + ModuleName + "/CodeGenerationTemplates/"); if (!Directory.Exists(moduleControllersPath)) { Directory.CreateDirectory(moduleControllersPath); } if (File.Exists(controllerPath)) { Context.Output.WriteLine(T("Controller {0} already exists in target Module {1}.", controllerName, moduleName)); return; } string controllerText = File.ReadAllText(templatesPath + "Controller.txt"); controllerText = controllerText.Replace("$$ModuleName$$", moduleName); controllerText = controllerText.Replace("$$ControllerName$$", controllerName); File.WriteAllText(controllerPath, controllerText); string projectFileText = File.ReadAllText(moduleCsProjPath); // The string searches in solution/project files can be made aware of comment lines. if (projectFileText.Contains("\r\n ", "Controllers\\" + controllerName + ".cs"); projectFileText = projectFileText.Insert(projectFileText.LastIndexOf("\r\n \r\n \r\n ", "Controllers\\" + controllerName + ".cs"); projectFileText = projectFileText.Insert(projectFileText.LastIndexOf(""), itemGroupReference); } File.WriteAllText(moduleCsProjPath, projectFileText); Context.Output.WriteLine(T("Controller {0} created successfully in Module {1}", controllerName, moduleName)); TouchSolution(Context.Output); } private void IntegrateModule(string moduleName) { string projectGuid = Guid.NewGuid().ToString().ToUpper(); CreateFilesFromTemplates(moduleName, projectGuid); // The string searches in solution/project files can be made aware of comment lines. if (IncludeInSolution) { AddToSolution(Context.Output, moduleName, projectGuid, "Modules", SolutionDirectoryModules); } } private void IntegrateTheme(string themeName, string baseTheme) { CreateThemeFromTemplates(Context.Output, themeName, baseTheme, CreateProject ? Guid.NewGuid().ToString().ToUpper() : null, IncludeInSolution); } private void CreateFilesFromTemplates(string moduleName, string projectGuid) { string modulePath = HostingEnvironment.MapPath("~/Modules/" + moduleName + "/"); string propertiesPath = modulePath + "Properties"; var content = new HashSet(); var folders = new HashSet(); var compile = new HashSet(); foreach (var folder in _moduleDirectories) { Directory.CreateDirectory(modulePath + folder); if (!String.IsNullOrEmpty(folder)) { folders.Add(modulePath + folder); } } File.WriteAllText(modulePath + "Assets.json", File.ReadAllText(_codeGenTemplatePath + "ModuleAssetsJson.txt")); content.Add(modulePath + "Assets.json"); File.WriteAllText(modulePath + "Assets\\Styles.less", File.ReadAllText(_codeGenTemplatePath + "ModuleStylesLess.txt")); content.Add(modulePath + "Assets\\Styles.less"); File.WriteAllText(modulePath + "Styles\\Styles.css", File.ReadAllText(_codeGenTemplatePath + "ModuleStylesCss.txt")); content.Add(modulePath + "Styles\\Styles.css"); File.WriteAllText(modulePath + "Styles\\Styles.min.css", File.ReadAllText(_codeGenTemplatePath + "ModuleStylesMinCss.txt")); content.Add(modulePath + "Styles\\Styles.min.css"); File.WriteAllText(modulePath + "packages.config", File.ReadAllText(_codeGenTemplatePath + "ModulePackagesConfig.txt")); content.Add(modulePath + "packages.config"); File.WriteAllText(modulePath + "Web.config", File.ReadAllText(_codeGenTemplatePath + "ModuleRootWebConfig.txt")); content.Add(modulePath + "Web.config"); File.WriteAllText(modulePath + "Scripts\\Web.config", File.ReadAllText(_codeGenTemplatePath + "StaticFilesWebConfig.txt")); content.Add(modulePath + "Scripts\\Web.config"); File.WriteAllText(modulePath + "Styles\\Web.config", File.ReadAllText(_codeGenTemplatePath + "StaticFilesWebConfig.txt")); content.Add(modulePath + "Styles\\Web.config"); string templateText = File.ReadAllText(_codeGenTemplatePath + "ModuleManifest.txt"); templateText = templateText.Replace("$$ModuleName$$", moduleName); File.WriteAllText(modulePath + "Module.txt", templateText, System.Text.Encoding.UTF8); content.Add(modulePath + "Module.txt"); templateText = File.ReadAllText(_codeGenTemplatePath + "ModuleAssemblyInfo.txt"); templateText = templateText.Replace("$$ModuleName$$", moduleName); templateText = templateText.Replace("$$ModuleTypeLibGuid$$", Guid.NewGuid().ToString()); File.WriteAllText(propertiesPath + "\\AssemblyInfo.cs", templateText); compile.Add(propertiesPath + "\\AssemblyInfo.cs"); var contentItemGroup = CreateProjectItemGroup(modulePath, content, folders); var compileItemGroup = CreateCompileItemGroup(modulePath, compile); File.WriteAllText(modulePath + moduleName + ".csproj", CreateCsProject(moduleName, projectGuid, contentItemGroup, compileItemGroup)); } private static string CreateCsProject(string projectName, string projectGuid, string contentItemGroup, string compileItemGroup) { string text = File.ReadAllText(_codeGenTemplatePath + "\\ModuleCsProj.txt"); text = text.Replace("$$ModuleName$$", projectName); text = text.Replace("$$ModuleProjectGuid$$", projectGuid); text = text.Replace("$$ContentIncludes$$", contentItemGroup ?? ""); text = text.Replace("$$CompileIncludes$$", compileItemGroup ?? ""); text = text.Replace("$$OrchardReferences$$", GetOrchardReferences()); return text; } private static string GetOrchardReferences(int modulesFolderRelativeDepth = 2) { var frameworkRelativeDepth = string.Join("\\", Enumerable.Repeat("..", modulesFolderRelativeDepth + 1)); var coreRelativeDepth = string.Join("\\", Enumerable.Repeat("..", modulesFolderRelativeDepth)); return IsSourceEnlistment() ? $@" {{2D1D92BB-4555-4CBE-8D0E-63563D6CE4C6}} Orchard.Framework $(MvcBuildViews) {{9916839C-39FC-4CEB-A5AF-89CA7E87119F}} Orchard.Core $(MvcBuildViews) " : @" False ..\..\bin\Orchard.Core.dll $(MvcBuildViews) False ..\..\bin\Orchard.Framework.dll $(MvcBuildViews) "; } private static bool IsSourceEnlistment() { return File.Exists(Directory.GetParent(_orchardWebProj).Parent.FullName + "\\Orchard.sln"); } private void CreateThemeFromTemplates(TextWriter output, string themeName, string baseTheme, string projectGuid, bool includeInSolution) { var themePath = HostingEnvironment.MapPath("~/Themes/" + themeName + "/"); var createdFiles = new HashSet(); var createdFolders = new HashSet(); // create directories foreach (var folderName in _themeDirectories) { var folder = themePath + folderName; Directory.CreateDirectory(folder); if (!String.IsNullOrEmpty(folderName)) { createdFolders.Add(folder); } } File.WriteAllText(themePath + "Assets.json", File.ReadAllText(_codeGenTemplatePath + "ModuleAssetsJson.txt")); createdFiles.Add(themePath + "Assets.json"); File.WriteAllText(themePath + "Assets\\Styles.less", File.ReadAllText(_codeGenTemplatePath + "ModuleStylesLess.txt")); createdFiles.Add(themePath + "Assets\\Styles.less"); File.WriteAllText(themePath + "Styles\\Styles.css", File.ReadAllText(_codeGenTemplatePath + "ModuleStylesCss.txt")); createdFiles.Add(themePath + "Styles\\Styles.css"); File.WriteAllText(themePath + "Styles\\Styles.min.css", File.ReadAllText(_codeGenTemplatePath + "ModuleStylesMinCss.txt")); createdFiles.Add(themePath + "Styles\\Styles.min.css"); File.WriteAllText(themePath + "Web.config", File.ReadAllText(_codeGenTemplatePath + "ModuleRootWebConfig.txt")); createdFiles.Add(themePath + "Web.config"); File.WriteAllText(themePath + "Scripts\\Web.config", File.ReadAllText(_codeGenTemplatePath + "StaticFilesWebConfig.txt")); createdFiles.Add(themePath + "Scripts\\Web.config"); File.WriteAllText(themePath + "Styles\\Web.config", File.ReadAllText(_codeGenTemplatePath + "StaticFilesWebConfig.txt")); createdFiles.Add(themePath + "Styles\\Web.config"); File.WriteAllText(themePath + "Content\\Web.config", File.ReadAllText(_codeGenTemplatePath + "StaticFilesWebConfig.txt")); createdFiles.Add(themePath + "Content\\Web.config"); var templateText = File.ReadAllText(_codeGenTemplatePath + "\\ThemeManifest.txt").Replace("$$ThemeName$$", themeName); if (string.IsNullOrEmpty(baseTheme)) { templateText = templateText.Replace("BaseTheme: $$BaseTheme$$\r\n", ""); } else { templateText = templateText.Replace("$$BaseTheme$$", baseTheme); } File.WriteAllText(themePath + "Theme.txt", templateText); createdFiles.Add(themePath + "Theme.txt"); File.WriteAllBytes(themePath + "Theme.png", File.ReadAllBytes(_codeGenTemplatePath + "Theme.png")); createdFiles.Add(themePath + "Theme.png"); File.WriteAllText(themePath + "Placement.info", File.ReadAllText(_codeGenTemplatePath + "Placement.info")); createdFiles.Add(themePath + "Placement.info"); // create new csproj for the theme if (projectGuid != null) { File.WriteAllText(themePath + "packages.config", File.ReadAllText(_codeGenTemplatePath + "ModulePackagesConfig.txt")); createdFiles.Add(themePath + "packages.config"); var itemGroup = CreateProjectItemGroup(themePath, createdFiles, createdFolders); string projectText = CreateCsProject(themeName, projectGuid, itemGroup, null); File.WriteAllText(themePath + "\\" + themeName + ".csproj", projectText); } if (includeInSolution) { if (projectGuid == null) { // include in solution but dont create a project: just add the references to Orchard.Themes project var itemGroup = CreateProjectItemGroup(HostingEnvironment.MapPath("~/Themes/"), createdFiles, createdFolders); AddFilesToOrchardThemesProject(output, itemGroup); TouchSolution(output); } else { // create a project (already done) and add it to the solution AddToSolution(output, themeName, projectGuid, "Themes", SolutionDirectoryThemes); } } } private void AddToSolution(TextWriter output, string projectName, string projectGuid, string containingFolder, string solutionFolderGuid) { if (!string.IsNullOrEmpty(projectGuid)) { var solutionPath = Directory.GetParent(_orchardWebProj).Parent.FullName + "\\Orchard.sln"; if (File.Exists(solutionPath)) { var projectReference = string.Format("EndProject\r\nProject(\"{{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}}\") = \"{0}\", \"Orchard.Web\\{2}\\{0}\\{0}.csproj\", \"{{{1}}}\"\r\n", projectName, projectGuid, containingFolder); var projectConfiguationPlatforms = string.Format("\t{{{0}}}.Debug|Any CPU.ActiveCfg = Debug|Any CPU\r\n\t\t{{{0}}}.Debug|Any CPU.Build.0 = Debug|Any CPU\r\n\t\t{{{0}}}.Release|Any CPU.ActiveCfg = Release|Any CPU\r\n\t\t{{{0}}}.Release|Any CPU.Build.0 = Release|Any CPU\r\n\t", projectGuid); var solutionText = File.ReadAllText(solutionPath); solutionText = solutionText.Insert(solutionText.LastIndexOf("EndProject\r\n"), projectReference); solutionText = AppendGlobalSection(solutionText, "ProjectConfigurationPlatforms", projectConfiguationPlatforms); solutionText = AppendGlobalSection(solutionText, "NestedProjects", "\t{" + projectGuid + "} = {" + solutionFolderGuid + "}\r\n\t"); File.WriteAllText(solutionPath, solutionText); TouchSolution(output); } } } private string AppendGlobalSection(string solutionText, string sectionName, string content) { var sectionStart = solutionText.IndexOf(string.Format("GlobalSection({0})", sectionName)); var sectionEnd = solutionText.IndexOf("EndGlobalSection", sectionStart); return solutionText.Insert(sectionEnd, content); } private static string CreateProjectItemGroup(string relativeFromPath, HashSet content, HashSet folders) { var contentInclude = ""; if (relativeFromPath != null && !relativeFromPath.EndsWith("\\", StringComparison.OrdinalIgnoreCase)) { relativeFromPath += "\\"; } else if (relativeFromPath == null) { relativeFromPath = ""; } if (content != null && content.Count > 0) { contentInclude = string.Join("\r\n", from file in content select " "); } if (folders != null && folders.Count > 0) { contentInclude += "\r\n" + string.Join("\r\n", from folder in folders select " "); } return string.Format(CultureInfo.InvariantCulture, "\r\n{0}\r\n \r\n ", contentInclude); } private static string CreateCompileItemGroup(string relativeFromPath, HashSet compile) { var compileInclude = ""; if (relativeFromPath != null && !relativeFromPath.EndsWith("\\", StringComparison.OrdinalIgnoreCase)) { relativeFromPath += "\\"; } else if (relativeFromPath == null) { relativeFromPath = ""; } if (compile != null && compile.Count > 0) { compileInclude = string.Join("\r\n", from file in compile select " "); } return string.Format(CultureInfo.InvariantCulture, "\r\n{0}\r\n \r\n ", compileInclude); } private void AddFilesToOrchardThemesProject(TextWriter output, string itemGroup) { if (!File.Exists(_orchardThemesProj)) { output.WriteLine(T("Warning: Orchard.Themes project file could not be found at {0}", _orchardThemesProj)); } else { var projectText = File.ReadAllText(_orchardThemesProj); // find where the first ItemGroup is after any References var refIndex = projectText.LastIndexOf("", refIndex); if (firstItemGroupIndex != -1) { projectText = projectText.Insert(firstItemGroupIndex, itemGroup); File.WriteAllText(_orchardThemesProj, projectText); return; } } output.WriteLine(T("Warning: Unable to modify Orchard.Themes project file at {0}", _orchardThemesProj)); } } private void TouchSolution(TextWriter output) { string rootWebProjectPath = HostingEnvironment.MapPath("~/Orchard.Web.csproj"); string solutionPath = Directory.GetParent(rootWebProjectPath).Parent.FullName + "\\Orchard.sln"; if (!File.Exists(solutionPath)) { output.WriteLine(T("Warning: Solution file could not be found at {0}", solutionPath)); return; } try { File.SetLastWriteTime(solutionPath, DateTime.Now); } catch { output.WriteLine(T("An unexpected error occurred while trying to refresh the Visual Studio solution. Please reload it.")); } } } }