├── .gitattributes ├── .gitignore ├── LICENSE ├── PublishSsh.sln ├── README.md └── src └── DotnetPublishSsh ├── DotnetPublishSsh.csproj ├── LocalFile.cs ├── Program.cs ├── PublishSshOptions.cs └── Uploader.cs /.gitattributes: -------------------------------------------------------------------------------- 1 | ############################################################################### 2 | # Set default behavior to automatically normalize line endings. 3 | ############################################################################### 4 | * text=auto 5 | 6 | ############################################################################### 7 | # Set default behavior for command prompt diff. 8 | # 9 | # This is need for earlier builds of msysgit that does not have it on by 10 | # default for csharp files. 11 | # Note: This is only used by command line 12 | ############################################################################### 13 | #*.cs diff=csharp 14 | 15 | ############################################################################### 16 | # Set the merge driver for project and solution files 17 | # 18 | # Merging from the command prompt will add diff markers to the files if there 19 | # are conflicts (Merging from VS is not affected by the settings below, in VS 20 | # the diff markers are never inserted). Diff markers may cause the following 21 | # file extensions to fail to load in VS. An alternative would be to treat 22 | # these files as binary and thus will always conflict and require user 23 | # intervention with every merge. To do so, just uncomment the entries below 24 | ############################################################################### 25 | #*.sln merge=binary 26 | #*.csproj merge=binary 27 | #*.vbproj merge=binary 28 | #*.vcxproj merge=binary 29 | #*.vcproj merge=binary 30 | #*.dbproj merge=binary 31 | #*.fsproj merge=binary 32 | #*.lsproj merge=binary 33 | #*.wixproj merge=binary 34 | #*.modelproj merge=binary 35 | #*.sqlproj merge=binary 36 | #*.wwaproj merge=binary 37 | 38 | ############################################################################### 39 | # behavior for image files 40 | # 41 | # image files are treated as binary by default. 42 | ############################################################################### 43 | #*.jpg binary 44 | #*.png binary 45 | #*.gif binary 46 | 47 | ############################################################################### 48 | # diff behavior for common document formats 49 | # 50 | # Convert binary document formats to text before diffing them. This feature 51 | # is only available from the command line. Turn it on by uncommenting the 52 | # entries below. 53 | ############################################################################### 54 | #*.doc diff=astextplain 55 | #*.DOC diff=astextplain 56 | #*.docx diff=astextplain 57 | #*.DOCX diff=astextplain 58 | #*.dot diff=astextplain 59 | #*.DOT diff=astextplain 60 | #*.pdf diff=astextplain 61 | #*.PDF diff=astextplain 62 | #*.rtf diff=astextplain 63 | #*.RTF diff=astextplain 64 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | ## Ignore Visual Studio temporary files, build results, and 2 | ## files generated by popular Visual Studio add-ons. 3 | 4 | # User-specific files 5 | *.suo 6 | *.user 7 | *.userosscache 8 | *.sln.docstates 9 | 10 | # User-specific files (MonoDevelop/Xamarin Studio) 11 | *.userprefs 12 | 13 | # Build results 14 | [Dd]ebug/ 15 | [Dd]ebugPublic/ 16 | [Rr]elease/ 17 | [Rr]eleases/ 18 | [Xx]64/ 19 | [Xx]86/ 20 | [Bb]uild/ 21 | bld/ 22 | [Bb]in/ 23 | [Oo]bj/ 24 | 25 | # Visual Studio 2015 cache/options directory 26 | .vs/ 27 | # Uncomment if you have tasks that create the project's static files in wwwroot 28 | #wwwroot/ 29 | 30 | # MSTest test Results 31 | [Tt]est[Rr]esult*/ 32 | [Bb]uild[Ll]og.* 33 | 34 | # NUNIT 35 | *.VisualState.xml 36 | TestResult.xml 37 | 38 | # Build Results of an ATL Project 39 | [Dd]ebugPS/ 40 | [Rr]eleasePS/ 41 | dlldata.c 42 | 43 | # DNX 44 | project.lock.json 45 | artifacts/ 46 | 47 | *_i.c 48 | *_p.c 49 | *_i.h 50 | *.ilk 51 | *.meta 52 | *.obj 53 | *.pch 54 | *.pdb 55 | *.pgc 56 | *.pgd 57 | *.rsp 58 | *.sbr 59 | *.tlb 60 | *.tli 61 | *.tlh 62 | *.tmp 63 | *.tmp_proj 64 | *.log 65 | *.vspscc 66 | *.vssscc 67 | .builds 68 | *.pidb 69 | *.svclog 70 | *.scc 71 | 72 | # Chutzpah Test files 73 | _Chutzpah* 74 | 75 | # Visual C++ cache files 76 | ipch/ 77 | *.aps 78 | *.ncb 79 | *.opendb 80 | *.opensdf 81 | *.sdf 82 | *.cachefile 83 | *.VC.db 84 | 85 | # Visual Studio profiler 86 | *.psess 87 | *.vsp 88 | *.vspx 89 | *.sap 90 | 91 | # TFS 2012 Local Workspace 92 | $tf/ 93 | 94 | # Guidance Automation Toolkit 95 | *.gpState 96 | 97 | # ReSharper is a .NET coding add-in 98 | _ReSharper*/ 99 | *.[Rr]e[Ss]harper 100 | *.DotSettings.user 101 | 102 | # JustCode is a .NET coding add-in 103 | .JustCode 104 | 105 | # TeamCity is a build add-in 106 | _TeamCity* 107 | 108 | # DotCover is a Code Coverage Tool 109 | *.dotCover 110 | 111 | # NCrunch 112 | _NCrunch_* 113 | .*crunch*.local.xml 114 | nCrunchTemp_* 115 | 116 | # MightyMoose 117 | *.mm.* 118 | AutoTest.Net/ 119 | 120 | # Web workbench (sass) 121 | .sass-cache/ 122 | 123 | # Installshield output folder 124 | [Ee]xpress/ 125 | 126 | # DocProject is a documentation generator add-in 127 | DocProject/buildhelp/ 128 | DocProject/Help/*.HxT 129 | DocProject/Help/*.HxC 130 | DocProject/Help/*.hhc 131 | DocProject/Help/*.hhk 132 | DocProject/Help/*.hhp 133 | DocProject/Help/Html2 134 | DocProject/Help/html 135 | 136 | # Click-Once directory 137 | publish/ 138 | 139 | # Publish Web Output 140 | *.[Pp]ublish.xml 141 | *.azurePubxml 142 | 143 | # TODO: Un-comment the next line if you do not want to checkin 144 | # your web deploy settings because they may include unencrypted 145 | # passwords 146 | #*.pubxml 147 | *.publishproj 148 | 149 | # NuGet Packages 150 | *.nupkg 151 | # The packages folder can be ignored because of Package Restore 152 | **/packages/* 153 | # except build/, which is used as an MSBuild target. 154 | !**/packages/build/ 155 | # Uncomment if necessary however generally it will be regenerated when needed 156 | #!**/packages/repositories.config 157 | # NuGet v3's project.json files produces more ignoreable files 158 | *.nuget.props 159 | *.nuget.targets 160 | 161 | # Microsoft Azure Build Output 162 | csx/ 163 | *.build.csdef 164 | 165 | # Microsoft Azure Emulator 166 | ecf/ 167 | rcf/ 168 | 169 | # Windows Store app package directory 170 | AppPackages/ 171 | BundleArtifacts/ 172 | 173 | # Visual Studio cache files 174 | # files ending in .cache can be ignored 175 | *.[Cc]ache 176 | # but keep track of directories ending in .cache 177 | !*.[Cc]ache/ 178 | 179 | # Others 180 | ClientBin/ 181 | [Ss]tyle[Cc]op.* 182 | ~$* 183 | *~ 184 | *.dbmdl 185 | *.dbproj.schemaview 186 | *.pfx 187 | *.publishsettings 188 | node_modules/ 189 | orleans.codegen.cs 190 | 191 | # RIA/Silverlight projects 192 | Generated_Code/ 193 | 194 | # Backup & report files from converting an old project file 195 | # to a newer Visual Studio version. Backup files are not needed, 196 | # because we have git ;-) 197 | _UpgradeReport_Files/ 198 | Backup*/ 199 | UpgradeLog*.XML 200 | UpgradeLog*.htm 201 | 202 | # SQL Server files 203 | *.mdf 204 | *.ldf 205 | 206 | # Business Intelligence projects 207 | *.rdl.data 208 | *.bim.layout 209 | *.bim_*.settings 210 | 211 | # Microsoft Fakes 212 | FakesAssemblies/ 213 | 214 | # GhostDoc plugin setting file 215 | *.GhostDoc.xml 216 | 217 | # Node.js Tools for Visual Studio 218 | .ntvs_analysis.dat 219 | 220 | # Visual Studio 6 build log 221 | *.plg 222 | 223 | # Visual Studio 6 workspace options file 224 | *.opt 225 | 226 | # Visual Studio LightSwitch build output 227 | **/*.HTMLClient/GeneratedArtifacts 228 | **/*.DesktopClient/GeneratedArtifacts 229 | **/*.DesktopClient/ModelManifest.xml 230 | **/*.Server/GeneratedArtifacts 231 | **/*.Server/ModelManifest.xml 232 | _Pvt_Extensions 233 | 234 | # LightSwitch generated files 235 | GeneratedArtifacts/ 236 | ModelManifest.xml 237 | 238 | # Paket dependency manager 239 | .paket/paket.exe 240 | 241 | # FAKE - F# Make 242 | .fake/ 243 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining a copy 4 | of this software and associated documentation files (the "Software"), to deal 5 | in the Software without restriction, including without limitation the rights 6 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 7 | copies of the Software, and to permit persons to whom the Software is 8 | furnished to do so, subject to the following conditions: 9 | 10 | The above copyright notice and this permission notice shall be included in all 11 | copies or substantial portions of the Software. 12 | 13 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 14 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 15 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 16 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 17 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 18 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 19 | SOFTWARE. 20 | -------------------------------------------------------------------------------- /PublishSsh.sln: -------------------------------------------------------------------------------- 1 |  2 | Microsoft Visual Studio Solution File, Format Version 12.00 3 | # Visual Studio 15 4 | VisualStudioVersion = 15.0.27130.2036 5 | MinimumVisualStudioVersion = 10.0.40219.1 6 | Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "src", "src", "{A0782477-557D-49FD-AD75-F9FAE287B10F}" 7 | EndProject 8 | Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{EE0D426E-738B-4664-935E-526A2F865AC2}" 9 | ProjectSection(SolutionItems) = preProject 10 | README.md = README.md 11 | EndProjectSection 12 | EndProject 13 | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "DotnetPublishSsh", "src\DotnetPublishSsh\DotnetPublishSsh.csproj", "{DE2662B1-F371-4072-BFC7-6C5F175C7ECE}" 14 | EndProject 15 | Global 16 | GlobalSection(SolutionConfigurationPlatforms) = preSolution 17 | Debug|Any CPU = Debug|Any CPU 18 | Release|Any CPU = Release|Any CPU 19 | EndGlobalSection 20 | GlobalSection(ProjectConfigurationPlatforms) = postSolution 21 | {DE2662B1-F371-4072-BFC7-6C5F175C7ECE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU 22 | {DE2662B1-F371-4072-BFC7-6C5F175C7ECE}.Debug|Any CPU.Build.0 = Debug|Any CPU 23 | {DE2662B1-F371-4072-BFC7-6C5F175C7ECE}.Release|Any CPU.ActiveCfg = Release|Any CPU 24 | {DE2662B1-F371-4072-BFC7-6C5F175C7ECE}.Release|Any CPU.Build.0 = Release|Any CPU 25 | EndGlobalSection 26 | GlobalSection(SolutionProperties) = preSolution 27 | HideSolutionNode = FALSE 28 | EndGlobalSection 29 | GlobalSection(NestedProjects) = preSolution 30 | {DE2662B1-F371-4072-BFC7-6C5F175C7ECE} = {A0782477-557D-49FD-AD75-F9FAE287B10F} 31 | EndGlobalSection 32 | GlobalSection(ExtensibilityGlobals) = postSolution 33 | SolutionGuid = {4B0DE057-7A0A-428A-95AE-1C1CC025053A} 34 | EndGlobalSection 35 | EndGlobal 36 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # dotnet-publish-ssh 2 | 3 | Simple publish your .Net Core application to linux server via SSH. 4 | 5 | # Usage 6 | 7 | * Add this to `csproj` file: 8 | ```XML 9 | 10 | 11 | 12 | ``` 13 | 14 | * Run `dotnet restore` 15 | * Run `dotnet publish-ssh` with options: 16 | ``` 17 | Usage: dotnet publish-ssh [arguments] [options] 18 | Arguments and options are the same as for `dotnet publish` 19 | SSH specific options: 20 | --ssh-host * Host address 21 | --ssh-port Host port 22 | --ssh-user * User name 23 | --ssh-password Password 24 | --ssh-keyfile Private OpenSSH key file 25 | --ssh-path * Publish path on remote server 26 | (*) required 27 | ``` 28 | 29 | # Example 30 | 31 | `dotnet publish-ssh --ssh-host 10.0.0.1 --ssh-port 22 --ssh-user root --ssh-password secret --ssh-path /var/www/site` 32 | 33 | # TODO 34 | 35 | - [x] Just works 36 | - [x] Password authentication 37 | - [x] Private key file authentication 38 | - [ ] Don't upload unmodified files (checksum) 39 | - [ ] Config file 40 | - [ ] Pre/post publish hooks on remote server 41 | -------------------------------------------------------------------------------- /src/DotnetPublishSsh/DotnetPublishSsh.csproj: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Publish .Net Core app to remote server via SSH 5 | dotnet-publish-ssh 6 | 0.1.0 7 | net461;netcoreapp1.0;netcoreapp1.1;netcoreapp2.0 8 | dotnet-publish-ssh 9 | Exe 10 | DotnetPublishSsh 11 | publish;deploy;ssh 12 | https://github.com/albekov/dotnet-publish-ssh 13 | git 14 | https://github.com/albekov/dotnet-publish-ssh 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | -------------------------------------------------------------------------------- /src/DotnetPublishSsh/LocalFile.cs: -------------------------------------------------------------------------------- 1 | using System; 2 | 3 | namespace DotnetPublishSsh 4 | { 5 | internal sealed class LocalFile 6 | { 7 | public LocalFile(string localPath, string fileName) 8 | { 9 | FileName = fileName; 10 | RelativeName = new Uri(localPath).MakeRelativeUri(new Uri(fileName)).OriginalString; 11 | } 12 | 13 | public string FileName { get; set; } 14 | 15 | public string RelativeName { get; set; } 16 | } 17 | } -------------------------------------------------------------------------------- /src/DotnetPublishSsh/Program.cs: -------------------------------------------------------------------------------- 1 | using System; 2 | using System.Collections.Generic; 3 | using System.Diagnostics; 4 | using System.IO; 5 | using System.Linq; 6 | 7 | namespace DotnetPublishSsh 8 | { 9 | internal sealed class Program 10 | { 11 | public static void Main(string[] args) 12 | { 13 | var options = PublishSshOptions.ParseArgs(args); 14 | if (options.PrintHelp) 15 | { 16 | PrintHelp(); 17 | return; 18 | } 19 | 20 | PrepareOptions(options); 21 | 22 | var arguments = string.Join(" ", options.Args); 23 | 24 | if (!PublishLocal(arguments)) 25 | return; 26 | 27 | var path = options.Path; 28 | var localPath = options.LocalPath; 29 | 30 | if (!path.EndsWith("/")) path = path + "/"; 31 | 32 | localPath = Path.GetFullPath(localPath) + Path.DirectorySeparatorChar; 33 | 34 | var localFiles = GetLocalFiles(localPath); 35 | 36 | Console.WriteLine(); 37 | Console.WriteLine($"Uploading {localFiles.Count} files to {options.User}@{options.User}:{options.Port}{options.Path}"); 38 | 39 | try 40 | { 41 | var uploader = new Uploader(options); 42 | 43 | uploader.UploadFiles(path, localFiles); 44 | } 45 | catch (Exception ex) 46 | { 47 | Console.WriteLine($"Error uploading files to server: {ex.Message}"); 48 | } 49 | Directory.Delete(localPath, true); 50 | } 51 | 52 | private static void PrepareOptions(PublishSshOptions options) 53 | { 54 | if (string.IsNullOrEmpty(options.LocalPath)) 55 | { 56 | var tempPath = Path.Combine(Path.GetTempPath(), $"publish.{Guid.NewGuid()}"); 57 | Directory.CreateDirectory(tempPath); 58 | options.LocalPath = tempPath; 59 | } 60 | 61 | options.Args = options.Args.Concat(new[] {"-o", options.LocalPath}).ToArray(); 62 | } 63 | 64 | private static bool PublishLocal(string arguments) 65 | { 66 | Console.WriteLine($"Starting `dotnet {arguments}`"); 67 | 68 | var info = new ProcessStartInfo 69 | { 70 | FileName = "dotnet", 71 | Arguments = "publish " + arguments 72 | }; 73 | 74 | var process = Process.Start(info); 75 | process.WaitForExit(); 76 | var exitCode = process.ExitCode; 77 | 78 | Console.WriteLine($"dotnet publish exited with code {exitCode}"); 79 | 80 | return exitCode == 0; 81 | } 82 | 83 | private static List GetLocalFiles(string localPath) 84 | { 85 | var localFiles = Directory 86 | .EnumerateFiles(localPath, "*.*", SearchOption.AllDirectories) 87 | .Select(f => new LocalFile(localPath, f)) 88 | .ToList(); 89 | return localFiles; 90 | } 91 | 92 | private static void PrintHelp() 93 | { 94 | Console.WriteLine("Publish to remote server via SSH"); 95 | Console.WriteLine(); 96 | Console.WriteLine("Usage: dotnet publish-ssh [arguments] [options]"); 97 | Console.WriteLine(); 98 | Console.WriteLine("Arguments and options are the same as for `dotnet publish`"); 99 | Console.WriteLine(); 100 | Console.WriteLine("SSH specific options:"); 101 | Console.WriteLine(" --ssh-host * Host address"); 102 | Console.WriteLine(" --ssh-port Host port"); 103 | Console.WriteLine(" --ssh-user * User name"); 104 | Console.WriteLine(" --ssh-password Password"); 105 | Console.WriteLine(" --ssh-keyfile Private OpenSSH key file"); 106 | Console.WriteLine(" --ssh-path * Publish path on remote server"); 107 | Console.WriteLine(); 108 | } 109 | } 110 | } -------------------------------------------------------------------------------- /src/DotnetPublishSsh/PublishSshOptions.cs: -------------------------------------------------------------------------------- 1 | using System; 2 | using System.Linq; 3 | 4 | namespace DotnetPublishSsh 5 | { 6 | internal sealed class PublishSshOptions 7 | { 8 | public string Host { get; set; } 9 | public int Port { get; set; } = 22; 10 | public string User { get; set; } 11 | public string Password { get; set; } 12 | public string KeyFile { get; set; } 13 | public string Path { get; set; } 14 | public string LocalPath { get; set; } 15 | public string[] Args { get; set; } 16 | public bool PrintHelp { get; set; } 17 | 18 | public static PublishSshOptions ParseArgs(string[] args) 19 | { 20 | var options = new PublishSshOptions(); 21 | 22 | for (var idx = 0; idx < args.Length; idx++) 23 | { 24 | var arg = args[idx]; 25 | switch (arg) 26 | { 27 | case "--ssh-host": 28 | options.Host = GetValue(ref args, ref idx); 29 | break; 30 | case "--ssh-port": 31 | var value = GetValue(ref args, ref idx); 32 | options.Port = Convert.ToInt32(value); 33 | break; 34 | case "--ssh-user": 35 | options.User = GetValue(ref args, ref idx); 36 | break; 37 | case "--ssh-password": 38 | options.Password = GetValue(ref args, ref idx); 39 | break; 40 | case "--ssh-keyfile": 41 | options.KeyFile = GetValue(ref args, ref idx); 42 | break; 43 | case "--ssh-path": 44 | options.Path = GetValue(ref args, ref idx); 45 | break; 46 | case "-o": 47 | options.LocalPath = GetValue(ref args, ref idx); 48 | break; 49 | case "-?": 50 | case "-h": 51 | case "--help": 52 | options.PrintHelp = true; 53 | break; 54 | } 55 | } 56 | 57 | ValidateOptions(options); 58 | 59 | options.Args = args; 60 | 61 | return options; 62 | } 63 | 64 | private static void ValidateOptions(PublishSshOptions options) 65 | { 66 | if (string.IsNullOrEmpty(options.Host) || 67 | string.IsNullOrEmpty(options.User) || 68 | string.IsNullOrEmpty(options.Path)) 69 | options.PrintHelp = true; 70 | } 71 | 72 | private static string GetValue(ref string[] args, ref int idx) 73 | { 74 | if (args.Length <= idx + 1) 75 | throw new ArgumentException($"Missing value for option {args[idx]}"); 76 | 77 | var value = args[idx + 1]; 78 | 79 | args = args.Take(idx).Concat(args.Skip(idx + 2)).ToArray(); 80 | idx--; 81 | 82 | return value; 83 | } 84 | } 85 | } -------------------------------------------------------------------------------- /src/DotnetPublishSsh/Uploader.cs: -------------------------------------------------------------------------------- 1 | using System; 2 | using System.Collections.Generic; 3 | using System.IO; 4 | using System.Linq; 5 | using Renci.SshNet; 6 | 7 | namespace DotnetPublishSsh 8 | { 9 | internal sealed class Uploader 10 | { 11 | public char DirectorySeparator { get; set; } = '/'; 12 | 13 | private readonly ConnectionInfo _connectionInfo; 14 | private readonly HashSet _existingDirectories = new HashSet(); 15 | 16 | public Uploader(PublishSshOptions publishSshOptions) 17 | { 18 | _connectionInfo = CreateConnectionInfo(publishSshOptions); 19 | } 20 | 21 | private static ConnectionInfo CreateConnectionInfo(PublishSshOptions options) 22 | { 23 | var authenticationMethods = new List(); 24 | 25 | if (options.Password != null) 26 | authenticationMethods.Add( 27 | new PasswordAuthenticationMethod(options.User, options.Password)); 28 | 29 | if (options.KeyFile != null) 30 | authenticationMethods.Add( 31 | new PrivateKeyAuthenticationMethod(options.User, new PrivateKeyFile(options.KeyFile))); 32 | 33 | var connectionInfo = new ConnectionInfo( 34 | options.Host, 35 | options.Port, 36 | options.User, 37 | authenticationMethods.ToArray()); 38 | 39 | return connectionInfo; 40 | } 41 | 42 | public void UploadFiles(string path, ICollection localFiles) 43 | { 44 | //using (var client = new SshClient(_connectionInfo)) 45 | using (var ftp = new SftpClient(_connectionInfo)) 46 | { 47 | //client.Connect(); 48 | ftp.Connect(); 49 | 50 | foreach (var localFile in localFiles) 51 | { 52 | UploadFile(localFile, ftp, path); 53 | } 54 | } 55 | Console.WriteLine($"Uploaded {localFiles.Count} files."); 56 | } 57 | 58 | private void UploadFile(LocalFile localFile, SftpClient ftp, string path) 59 | { 60 | Console.WriteLine($"Uploading {localFile.RelativeName}"); 61 | using (var stream = File.OpenRead(localFile.FileName)) 62 | { 63 | var filePath = localFile.RelativeName.Replace(Path.DirectorySeparatorChar, DirectorySeparator); 64 | 65 | var fullPath = path + filePath; 66 | 67 | EnsureDirExists(ftp, fullPath); 68 | 69 | ftp.UploadFile(stream, fullPath, true); 70 | } 71 | } 72 | 73 | private void EnsureDirExists(SftpClient ftp, string path) 74 | { 75 | var parts = path.Split(new[] {DirectorySeparator}, StringSplitOptions.RemoveEmptyEntries) 76 | .Where(p => !string.IsNullOrEmpty(p)) 77 | .ToList(); 78 | 79 | if (!path.EndsWith(DirectorySeparator.ToString())) 80 | parts = parts.Take(parts.Count - 1).ToList(); 81 | 82 | CreateDir(ftp, parts); 83 | } 84 | 85 | private void CreateDir(SftpClient ftp, ICollection parts, bool noCheck = false) 86 | { 87 | if (parts.Any()) 88 | { 89 | var path = Combine(parts); 90 | var parent = parts.Take(parts.Count - 1).ToList(); 91 | 92 | if (noCheck || ftp.Exists(path)) 93 | { 94 | CreateDir(ftp, parent, true); 95 | } 96 | else 97 | { 98 | CreateDir(ftp, parent); 99 | ftp.CreateDirectory(path); 100 | } 101 | 102 | _existingDirectories.Add(path); 103 | } 104 | } 105 | 106 | private string Combine(ICollection parts) 107 | { 108 | var path = DirectorySeparator + 109 | string.Join(DirectorySeparator.ToString(), parts) + 110 | (parts.Any() ? DirectorySeparator.ToString() : ""); 111 | return path; 112 | } 113 | } 114 | } --------------------------------------------------------------------------------