When i publish with dotnet
cli, the output is:
publish: Published to ../output/project1
Published 1/1 projects successfully
Which indicates i could publish multiple projects at te same time (why else say: 1/1)
so i tried this.
dotnet publish project1 project2 -o ../output
I expect 2 projects to be published in this output folder (each in there own folder, having the same name as the project folder).
Unrecognized command or argument project2
.NET Command Line Tools (1.0.0-preview2-1-003155)
Product Information:
Version: 1.0.0-preview2-1-003155
Commit SHA-1 hash: d7b0190bd4
Runtime Environment:
OS Name: Windows
OS Version: 10.0.14393
OS Platform: Windows
RID: win10-x64
@blackdwarf I agree with @joelharkes's assertion that the experience in preview2.1 was confusing. Thoughts? Note, this is a bug on the PJ CLI so it may not apply directly, but what is our intent on multi-project [sln?] publish?
@piotrpMSFT I think multi publish should be considered.
Our architecture is made up of 2 kestrel console apps and 3 other console applications.
now i have to execute 5 times dotnet publish
. 5 times going over library projects and all executing in synchronous order (otherwise i don't know if it will run into locking problems with trying to build the same project). This is a lot of overhead. if this is managed by the CLI, projects could be run asynchronously, speeding up the build performance with a big factor.
@blackdwarf will propose the intention change here.
For publish, I don't think publishing from a solution falls naturally because it is highly dependent on the solution architecture and the overall type of projects that are contained within. For some solutions, like the one @joelharkes mentions, it may make sense. For others that I saw and discussed as I was thinking about this, the expected behavior would be exactly the opposite, that is, publish
would be expected not to work on SLN files as that would produce an output that would not be wanted.
There are also many nuances that need to be thought of when thinking on this, mostly w.r.t higher-level tooling that will also consume this target and how does running that target on the solution works for their UX-es and capabilities.
So, for V1 at least, I don't see us supporting this behavior.
This is can be easy be done with powershell script.
You just have to create for each project build/publish script, and then execute it within loop.
This way I build/publish/create nuget/push nuget with one command for more than 30 projects in one solution.
This code could be in some PublishMultipleProjects.ps1 script.
You can extend this by adding parms what to include and so on...
````Powershell
Push-Location $PSScriptRoot
$files = get-childitem -Recurse -Include "Build.ps1"
foreach ($file in $files)
{
# Execute each Build.ps1 script
& $file
}
Push-Location $PSScriptRoot
@neman That is exactly what you want to avoid for 2 main reasons:
I made a script that builds the dependency trees for all projects and then executes builds in parallel (Promises nodejs) according to the builds. I got a major decrease in build/publish time (on a multi core environment).
Too bad though our build server agents only use 1 processor per agent but, but locally for developers its still better this way.
I've managed to bring down build+test time by using msbuild scripts like this (even in non-parallel builds): https://gist.github.com/dasMulli/69f5303aa79a8cd4060e44891c90fd2d
Essentially, making a single build.proj
file that calls publish on others via <MSBuild/>
tasks (=> dotnet msbuild build.proj
.
@dasMulli looks good, but ContinueOnError="ErrorAndContinue"
this seems very dangerous/error-prone? are you sure dependencies are always build before projects higher in the tree are build?
I've used ErrorAndContinue
to make sure it at least tries to test all test projects. The target will still return errors but ensures that all tasks are executed. Dependencies are always built before individual projects and msbuild should already handle concurrency situations inside a single msbuild invocation. For lots of test projects, it's probably even faster to build a solution and then running tests in parallel using VSTestNoBuild=true
as additional parameter.
We are planning our migration to csproj.
I seem to have the same problem here.
I want to publish 5 startup projects at the same time not needing to rebuild library projects more than once. But still there seems no possiblity?
if i dotnet publish solution.sln
it puts all the 5 projects into 1 folder.
If i run dotnet publish project1.csproj
it publishes 1 project but first builds all the project refernces.
Could there be an option made to either specify separate publish projects in the solution file? Or an option: --no-build
like on dotnet pack to publish a project without rebuilding it?
@blackdwarf is there anything on the roadmap for this feature? it could greatly speed up our build process.
@joelharkes sorry, but I'm not with Microsoft anymore so I really don't know what the roadmap is like these days. @livarcocc @richlander and @bleroy could help with this though, I would think.
FWIW, when we took a look at publishing based on the SLN file, I remember that there was a big split on whether the behavior could be made sane enough for most use cases to work. Some people were very for it (similar to dotnet restore
or dotnet build
), while others thought it was exactly the wrong thing to do. Add to it the options for redirecting output (--output
) and you have a nice can 'o' worms there.
Your idea of having a --no-build
option on dotnet publish
does sound useful, so I guess a separate issue could be made for that, but that would also have to go into the dotnet/sdk repo IINM.
having the same behaviour here, I have two projects to be published and "dot net" puts both on the same page 👎
cc @KathleenDollard
@gandarez are you publishing a solution or publishing with -o?
@livarcocc
I'm publishing through vsts using -o
This issue is still open, meaning we haven't said no. But we also do not have it on the roadmap as far as I know. I believe it needs some thought to get right.
@ganderez - I am a bit confused about your scenario. Can you send more information so we can figure out if this is a CLI issue or a VSTS issue?
@KathleenDollard
The solution contains two web api core projects and I'm using dot net publish to publish them. But the cli put every single project to the same root output directory.
The MSBUILD does not have that behaviour, it groups each project in its respective folder.
@gandarez when you say "The MSBUILD does not have that behaviour", how have you been using msbuild? the msbuild equivalent of dotnet publish -o foo
would be msbuild /t:Publish /p:PublishDir=foo
. If you have been using /p:DeployOnBuild=true
with a publish profile, that is a different way of publishing specific to asp.net and asp.net core and may behave differently.
It should also be noted that if you pass a relative path to the -o
parameter, it is always interpreted relative to the individual csproj file, not relative to the path the dotnet publish
command is invoked from.
@gandarez can you answer @dasMulli questions above so that we can understand your scenario better?
One solution that seems simple to me would be to have a variable that references the current project being built/published so one can have dotnet publish -o foo\$(Build.CurrentProject)
or msbuild /t:Publish /p:PublishDir=foo\$(Build.CurrentProject)
if one wants separate folders for each publish in a solution.
Or maybe this already exists? I am not seeing it in the build variables documentation at any rate.
Has there been any progress on this issue, or do we know of any other way around this?
We publish several console apps to the same dir and distribute that. ILLink.Tasks runs as part of publish. Without the ability to publish several projects at once, how can the linker do its job properly?
Closing issue, there is probably an alternative by now.
@joelharkes I haven't found an alternative yet. Paket's multi-push is broken as well https://github.com/fsprojects/Paket/issues/3537
@joelharkes why was this closed? There is no alternative solution and nothing in this issue suggests there is.
Probably the workaround is to join projects into a solution and run dotnet publish on solution, this makes a simultaneous publish.
@Lanayx no this is exactly what doesn't work, and what brought me to this issue. Running dotnet publish -o outputPath
on the solution will not create a separate folder for each project, it will dump everything into the single folder outputPath
.
@mariusGundersen I use dotnet publish -c Release -o artifacts
and it publishes each project to the _artifacts_ folder that is situated at each project's location
That will result in the path to the output being ./{project}/artifacts
while what I would want is ./artifacts/{project}
. Unfortunately that is not possible
@mariusGundersen how do you expect artifacts to be situated there? Should they reflect the initial folder structure? If yes, then how is it better than the available option?
Compare it to how dotnet pack -o outputDir
works. It uses the name of the csproj for the name of the nupkg. Publish could work the same way, taking the name of the csproj and using it as the name of the directory that the content is published to.
Sorry guys, was not working in dotnet core for a long while so i though clean up my open issues im no longer working on (and might already be outdated).
I'll re-open it again for you guys
@mariusGundersen Still; this issue is about publishing multiple projects with a command, not to change the directory structure of publishes of a single .sln
file. I think it would be appropriate to file a new issue if you want another type of change, than what this thread is about.
Workaround: I started using https://github.com/eiriktsarpalis/snippets/tree/master/SlnTools to package multiple projects with FAKE.
Here is a workaround to publish multi projects in a solution
Directory example
📦src
┣ 📂Web
┃ ┗ 📜Web.csproj
┣ 📂WebApp2
┃ ┗ 📜WebApp2.csproj
┣ 📜publish.targets
┗ 📜sln.sln
PublishRootDir
or use command line args /p:PublishRootDir=somedir
<Project>
<PropertyGroup>
<PublishRootDir>..\..\bin</PublishRootDir>
</PropertyGroup>
<Target Name="PublishAssets" AfterTargets="Publish">
<ItemGroup>
<_PublishAssets Include="$(OutDir)\publish\**\*.*" />
</ItemGroup>
<Message Importance="High" Text="Publishing assets " />
<Copy SourceFiles="@(_PublishAssets)" DestinationFolder="$(PublishRootDir)\$(TargetName)\%(RecursiveDir)" SkipUnchangedFiles="true" />
</Target>
</Project>
<Import Project="$(MSBuildThisFileDirectory)..\publish.targets" />
src
😁This is a 3 year old issue that should be trivial to fix, so I'm not sure why this is still lingering around? Several suggested solutions have been presented and even the referencing issues to this one outline how this mistake came to be.
-o
append each project name (just like pack is doing) to create a sub folder for each project. It does not make sense to dump everything to a single folder when publishing a solution and I would argue that when it would, this feature would be the minority and you would want a flag to enable it as the amount of issues this can cause has been outlined in other issues.
For now, I have choosen to build tooling around my tooling and removed the -o
param and have instead wrote a script to recursively find all the publish folders generated under each project and to move them to a root level folder.
Most helpful comment
This is a 3 year old issue that should be trivial to fix, so I'm not sure why this is still lingering around? Several suggested solutions have been presented and even the referencing issues to this one outline how this mistake came to be.
-o
append each project name (just like pack is doing) to create a sub folder for each project.It does not make sense to dump everything to a single folder when publishing a solution and I would argue that when it would, this feature would be the minority and you would want a flag to enable it as the amount of issues this can cause has been outlined in other issues.
For now, I have choosen to build tooling around my tooling and removed the
-o
param and have instead wrote a script to recursively find all the publish folders generated under each project and to move them to a root level folder.