This is an add-in for Fody which lets you trim unused private/internal types during compile-time.
This project has a dependency to Emik.Morsels, if you are building this project, refer to its README first.
-
Install the NuGet packages
Fody
andAbsence.Fody
. InstallingFody
explicitly is needed to enable weaving.PM> Install-Package Fody PM> Install-Package Absence.Fody
-
Add the
PrivateAssets="all"
metadata attribute to the<PackageReference />
items ofFody
andAbsence.Fody
in your project file, so they won't be listed as dependencies. -
If you already have a
FodyWeavers.xml
file in the root directory of your project, add the<Absence />
tag there. This file will be created on the first build if it doesn't exist:
<Weavers>
<Absence />
</Weavers>
See Fody usage for general guidelines, and Fody Configuration for additional options.
You can add an Except
attribute with regex strings to exclude namespaces, types, or members, separated by any amount of whitespace:
<Weavers>
<Absence Except="DoNotTrimMe Do.Not.Trim.Me
DoNotTrimGenericsEither`1" />
</Weavers>
What you write:
public class Good1
{
public class Good2
{
public static void Good3<[UsedImplicitly] T>() { }
[UsedImplicitly]
static void Good4<T>() { }
static void Bad1() { }
}
[UsedImplicitly]
class Good5
{
public static void Good6() { }
[UsedImplicitly]
static void Good7() { }
static void Bad2() { }
}
class Bad3
{
public static void Bad4() { }
[UsedImplicitly]
static void Bad5() { }
static void Bad6() { }
}
}
[UsedImplicitly]
class Good8
{
public class Good9
{
public static void Good10()
{
Good15<int>();
_ = new Good16();
}
[UsedImplicitly]
static void Good11() { }
static void Bad7() { }
}
[UsedImplicitly]
class Good12
{
public static void Good13() { }
[UsedImplicitly]
static void Good14() { }
static void Bad8() { }
}
class Bad9
{
public static void Bad10() { }
[UsedImplicitly]
static void Bad11() { }
static void Bad12() { }
}
static void Good15<[UsedImplicitly] T>() { }
static void Bad13<[UsedImplicitly] T>() { }
sealed class Good16
{
internal static List<KeyValuePair<string, Good17>> s_field = [];
[Good18.Good19]
internal struct Good17
{
sealed class Good18 : Attribute
{
internal sealed class Good19 : Attribute;
}
}
}
}
class Bad14
{
public class Bad15
{
public static void Bad16() { }
[UsedImplicitly]
static void Bad17() { }
static void Bad18() { }
}
[UsedImplicitly]
class Bad19
{
public static void Bad20() { }
[UsedImplicitly]
static void Bad21() { }
static void Bad22() { }
}
class Bad23
{
public static void Bad24() { }
[UsedImplicitly]
static void Bad25() { }
static void Bad26() { }
}
}
What gets compiled:
public class Good1
{
public class Good2
{
public static void Good3<[UsedImplicitly] T>() { }
[UsedImplicitly]
static void Good4<T>() { }
}
[UsedImplicitly]
class Good5
{
public static void Good6() { }
[UsedImplicitly]
static void Good7() { }
}
}
[UsedImplicitly]
class Good8
{
public class Good9
{
public static void Good10()
{
Good15<int>();
_ = new Good16();
}
[UsedImplicitly]
static void Good11() { }
}
[UsedImplicitly]
class Good12
{
public static void Good13() { }
[UsedImplicitly]
static void Good14() { }
}
static void Good15<[UsedImplicitly] T>() { }
sealed class Good16
{
internal static List<KeyValuePair<string, Good17>> s_field = [];
[Good18.Good19]
internal struct Good17
{
sealed class Good18 : Attribute
{
internal sealed class Good19 : Attribute;
}
}
}
}
Issues and pull requests are welcome to help this repository be the best it can be.
This repository falls under the MPL-2 license.