Immutable builder and updaterImmutable Queue in Java using an Immutable StackBloch's Builder Pattern / UpdaterInitializing immutable objects with a nested builderAlternative to the Josh Bloch Builder pattern in C#Generic immutable object builderReally simple immutable classSelective updates to immutable typesImmutable type updater using a special constructor

What is a CirKle Word™?

Place of the time marker in a sentence

Was there ever a treaty between 2 entities with significantly different translations to the detriment of one party?

Are the players on the same team as the DM?

Compelling story with the world as a villain

Are there any elected officials in the U.S. who are not legislators, judges, or constitutional officers?

Nothing like a good ol' game of ModTen

Disambiguation of "nobis vobis" and "nobis nobis"

LeetCode: Group Anagrams C#

Why did MS-DOS applications built using Turbo Pascal fail to start with a division by zero error on faster systems?

Non-visual Computers - thoughts?

Round towards zero

Is there any way to keep a player from killing an NPC?

Prove your innocence

How to make Ubuntu support single display 5120x1440 resolution?

“T” in subscript in formulas

In an interdisciplinary context, is the argument of "imposter syndrome" really applicable?

How to gently end involvement with an online community?

Can a Rogue PC teach an NPC to perform Sneak Attack?

Would the Republic of Ireland and Northern Ireland be interested in reuniting?

How to prevent clipped screen edges on my TV, HDMI-connected?

What are some interesting features that are common cross-linguistically but don't exist in English?

Is a player able to change alignment midway through an adventure?

Why isn't "I've" a proper response?



Immutable builder and updater


Immutable Queue in Java using an Immutable StackBloch's Builder Pattern / UpdaterInitializing immutable objects with a nested builderAlternative to the Josh Bloch Builder pattern in C#Generic immutable object builderReally simple immutable classSelective updates to immutable typesImmutable type updater using a special constructor






.everyoneloves__top-leaderboard:empty,.everyoneloves__mid-leaderboard:empty,.everyoneloves__bot-mid-leaderboard:empty margin-bottom:0;








8












$begingroup$


There aren't enough questions about creating immutable objects... so why not try it again with another approach. This time, it's a builder that maps properties to constructor parameters. Properties are selected via the With method that also expects a new value for the parameter. Constructor parameters must match properties so it's primarily for DTOs that follow this pattern. Build tries to find that constructor and creates the object.



// I need this primarily for the left-join of optional parameters.
internal class IgnoreCase : IEquatable<string>

public string Value get; set;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase Value = value ;


public class ImmutableBuilder<T>

private readonly IList<(MemberExpression Selector, object Value)> _selectors = new List<(MemberExpression Selector, object Value)>();

public ImmutableBuilder<T> With<TProperty>(Expression<Func<T, TProperty>> selector, TProperty value)

_selectors.Add(((MemberExpression)selector.Body, value));
return this;


public T Build()

var ctors =
from ctor in typeof(T).GetConstructors()
let parameters = ctor.GetParameters()
// Join parameters and values by parameter order.
// The ctor requires them sorted but they might be initialized in any order.
let requiredParameterValues =
from parameter in parameters.Where(p => !p.IsOptional)
join selector in _selectors on (IgnoreCase)parameter.Name equals (IgnoreCase)selector.Selector.Member.Name
select selector.Value
// Get optional parameters if any.
let optionalParameterValues =
from parameter in parameters.Where(p => p.IsOptional)
join selector in _selectors on (IgnoreCase)parameter.Name equals (IgnoreCase)selector.Selector.Member.Name into s
from selector in s.DefaultIfEmpty()
select selector.Value
// Make sure all required parameters are specified.
where requiredParameterValues.Count() == parameters.Where(p => !p.IsOptional).Count()
select (ctor, requiredParameterValues, optionalParameterValues);

var theOne = ctors.Single();
return (T)theOne.ctor.Invoke(theOne.requiredParameterValues.Concat(theOne.optionalParameterValues).ToArray());



public static class Immutable<T>

public static ImmutableBuilder<T> Builder => new ImmutableBuilder<T>();



In case we already have an immutable type and want to modify it by changing just some values, the ImmutableHelper can be used. It also provides the With method expecting a new value. Then matches it and other properties with the constructor and uses origianl values as defaults and for the specified property the new value.



public static class ImmutableHelper

public static T With<T, TProperty>(this T obj, Expression<Func<T, TProperty>> selector, TProperty value)

var comparer = StringComparer.OrdinalIgnoreCase;

var propertyName = ((MemberExpression)selector.Body).Member.Name;
var properties = typeof(T).GetProperties();

var propertyNames =
properties
.Select(x => x.Name)
// A hash-set is convenient for name matching in the next step.
.ToImmutableHashSet(comparer);

// Find the constructor that matches property names.
var ctors =
from ctor in typeof(T).GetConstructors()
where propertyNames.IsProperSupersetOf(ctor.GetParameters().Select(x => x.Name))
select ctor;

var theOne = ctors.Single(); // There can be only one match.

var parameters =
from parameter in theOne.GetParameters()
join property in properties on (IgnoreCase)parameter.Name equals (IgnoreCase)property.Name
// They definitely match so no comparer is necessary.
// Use either the new value or the current one.
select property.Name == propertyName ? value : property.GetValue(obj);

return (T)theOne.Invoke(parameters.ToArray());




Example



This is how it can be used:



void Main()

var person =
Immutable<Person>
.Builder
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Build();

person.With(x => x.LastName, "Doe").Dump();


public class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";




What do you say? Crazy? Insane? I like it? Let's improve it?










share|improve this question











$endgroup$









  • 2




    $begingroup$
    This has a use case for building controllers and what not that have dozens of parameters. Does this also work with optional ctr arguments?
    $endgroup$
    – dfhwze
    9 hours ago






  • 1




    $begingroup$
    @dfhwze it doesn't... but it definitelly could... and should.
    $endgroup$
    – t3chb0t
    9 hours ago










  • $begingroup$
    Ok ... so let's improve it! :p
    $endgroup$
    – dfhwze
    9 hours ago










  • $begingroup$
    @dfhwze done! ;-] at least the main builder can do this now.
    $endgroup$
    – t3chb0t
    8 hours ago







  • 1




    $begingroup$
    I like the With extensions method that helps to create modified copies of mutable types. However I didn't get the use case of the ImmutableBuilder.. Why not just calling the constructor?
    $endgroup$
    – JanDotNet
    5 hours ago

















8












$begingroup$


There aren't enough questions about creating immutable objects... so why not try it again with another approach. This time, it's a builder that maps properties to constructor parameters. Properties are selected via the With method that also expects a new value for the parameter. Constructor parameters must match properties so it's primarily for DTOs that follow this pattern. Build tries to find that constructor and creates the object.



// I need this primarily for the left-join of optional parameters.
internal class IgnoreCase : IEquatable<string>

public string Value get; set;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase Value = value ;


public class ImmutableBuilder<T>

private readonly IList<(MemberExpression Selector, object Value)> _selectors = new List<(MemberExpression Selector, object Value)>();

public ImmutableBuilder<T> With<TProperty>(Expression<Func<T, TProperty>> selector, TProperty value)

_selectors.Add(((MemberExpression)selector.Body, value));
return this;


public T Build()

var ctors =
from ctor in typeof(T).GetConstructors()
let parameters = ctor.GetParameters()
// Join parameters and values by parameter order.
// The ctor requires them sorted but they might be initialized in any order.
let requiredParameterValues =
from parameter in parameters.Where(p => !p.IsOptional)
join selector in _selectors on (IgnoreCase)parameter.Name equals (IgnoreCase)selector.Selector.Member.Name
select selector.Value
// Get optional parameters if any.
let optionalParameterValues =
from parameter in parameters.Where(p => p.IsOptional)
join selector in _selectors on (IgnoreCase)parameter.Name equals (IgnoreCase)selector.Selector.Member.Name into s
from selector in s.DefaultIfEmpty()
select selector.Value
// Make sure all required parameters are specified.
where requiredParameterValues.Count() == parameters.Where(p => !p.IsOptional).Count()
select (ctor, requiredParameterValues, optionalParameterValues);

var theOne = ctors.Single();
return (T)theOne.ctor.Invoke(theOne.requiredParameterValues.Concat(theOne.optionalParameterValues).ToArray());



public static class Immutable<T>

public static ImmutableBuilder<T> Builder => new ImmutableBuilder<T>();



In case we already have an immutable type and want to modify it by changing just some values, the ImmutableHelper can be used. It also provides the With method expecting a new value. Then matches it and other properties with the constructor and uses origianl values as defaults and for the specified property the new value.



public static class ImmutableHelper

public static T With<T, TProperty>(this T obj, Expression<Func<T, TProperty>> selector, TProperty value)

var comparer = StringComparer.OrdinalIgnoreCase;

var propertyName = ((MemberExpression)selector.Body).Member.Name;
var properties = typeof(T).GetProperties();

var propertyNames =
properties
.Select(x => x.Name)
// A hash-set is convenient for name matching in the next step.
.ToImmutableHashSet(comparer);

// Find the constructor that matches property names.
var ctors =
from ctor in typeof(T).GetConstructors()
where propertyNames.IsProperSupersetOf(ctor.GetParameters().Select(x => x.Name))
select ctor;

var theOne = ctors.Single(); // There can be only one match.

var parameters =
from parameter in theOne.GetParameters()
join property in properties on (IgnoreCase)parameter.Name equals (IgnoreCase)property.Name
// They definitely match so no comparer is necessary.
// Use either the new value or the current one.
select property.Name == propertyName ? value : property.GetValue(obj);

return (T)theOne.Invoke(parameters.ToArray());




Example



This is how it can be used:



void Main()

var person =
Immutable<Person>
.Builder
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Build();

person.With(x => x.LastName, "Doe").Dump();


public class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";




What do you say? Crazy? Insane? I like it? Let's improve it?










share|improve this question











$endgroup$









  • 2




    $begingroup$
    This has a use case for building controllers and what not that have dozens of parameters. Does this also work with optional ctr arguments?
    $endgroup$
    – dfhwze
    9 hours ago






  • 1




    $begingroup$
    @dfhwze it doesn't... but it definitelly could... and should.
    $endgroup$
    – t3chb0t
    9 hours ago










  • $begingroup$
    Ok ... so let's improve it! :p
    $endgroup$
    – dfhwze
    9 hours ago










  • $begingroup$
    @dfhwze done! ;-] at least the main builder can do this now.
    $endgroup$
    – t3chb0t
    8 hours ago







  • 1




    $begingroup$
    I like the With extensions method that helps to create modified copies of mutable types. However I didn't get the use case of the ImmutableBuilder.. Why not just calling the constructor?
    $endgroup$
    – JanDotNet
    5 hours ago













8












8








8


1



$begingroup$


There aren't enough questions about creating immutable objects... so why not try it again with another approach. This time, it's a builder that maps properties to constructor parameters. Properties are selected via the With method that also expects a new value for the parameter. Constructor parameters must match properties so it's primarily for DTOs that follow this pattern. Build tries to find that constructor and creates the object.



// I need this primarily for the left-join of optional parameters.
internal class IgnoreCase : IEquatable<string>

public string Value get; set;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase Value = value ;


public class ImmutableBuilder<T>

private readonly IList<(MemberExpression Selector, object Value)> _selectors = new List<(MemberExpression Selector, object Value)>();

public ImmutableBuilder<T> With<TProperty>(Expression<Func<T, TProperty>> selector, TProperty value)

_selectors.Add(((MemberExpression)selector.Body, value));
return this;


public T Build()

var ctors =
from ctor in typeof(T).GetConstructors()
let parameters = ctor.GetParameters()
// Join parameters and values by parameter order.
// The ctor requires them sorted but they might be initialized in any order.
let requiredParameterValues =
from parameter in parameters.Where(p => !p.IsOptional)
join selector in _selectors on (IgnoreCase)parameter.Name equals (IgnoreCase)selector.Selector.Member.Name
select selector.Value
// Get optional parameters if any.
let optionalParameterValues =
from parameter in parameters.Where(p => p.IsOptional)
join selector in _selectors on (IgnoreCase)parameter.Name equals (IgnoreCase)selector.Selector.Member.Name into s
from selector in s.DefaultIfEmpty()
select selector.Value
// Make sure all required parameters are specified.
where requiredParameterValues.Count() == parameters.Where(p => !p.IsOptional).Count()
select (ctor, requiredParameterValues, optionalParameterValues);

var theOne = ctors.Single();
return (T)theOne.ctor.Invoke(theOne.requiredParameterValues.Concat(theOne.optionalParameterValues).ToArray());



public static class Immutable<T>

public static ImmutableBuilder<T> Builder => new ImmutableBuilder<T>();



In case we already have an immutable type and want to modify it by changing just some values, the ImmutableHelper can be used. It also provides the With method expecting a new value. Then matches it and other properties with the constructor and uses origianl values as defaults and for the specified property the new value.



public static class ImmutableHelper

public static T With<T, TProperty>(this T obj, Expression<Func<T, TProperty>> selector, TProperty value)

var comparer = StringComparer.OrdinalIgnoreCase;

var propertyName = ((MemberExpression)selector.Body).Member.Name;
var properties = typeof(T).GetProperties();

var propertyNames =
properties
.Select(x => x.Name)
// A hash-set is convenient for name matching in the next step.
.ToImmutableHashSet(comparer);

// Find the constructor that matches property names.
var ctors =
from ctor in typeof(T).GetConstructors()
where propertyNames.IsProperSupersetOf(ctor.GetParameters().Select(x => x.Name))
select ctor;

var theOne = ctors.Single(); // There can be only one match.

var parameters =
from parameter in theOne.GetParameters()
join property in properties on (IgnoreCase)parameter.Name equals (IgnoreCase)property.Name
// They definitely match so no comparer is necessary.
// Use either the new value or the current one.
select property.Name == propertyName ? value : property.GetValue(obj);

return (T)theOne.Invoke(parameters.ToArray());




Example



This is how it can be used:



void Main()

var person =
Immutable<Person>
.Builder
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Build();

person.With(x => x.LastName, "Doe").Dump();


public class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";




What do you say? Crazy? Insane? I like it? Let's improve it?










share|improve this question











$endgroup$




There aren't enough questions about creating immutable objects... so why not try it again with another approach. This time, it's a builder that maps properties to constructor parameters. Properties are selected via the With method that also expects a new value for the parameter. Constructor parameters must match properties so it's primarily for DTOs that follow this pattern. Build tries to find that constructor and creates the object.



// I need this primarily for the left-join of optional parameters.
internal class IgnoreCase : IEquatable<string>

public string Value get; set;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase Value = value ;


public class ImmutableBuilder<T>

private readonly IList<(MemberExpression Selector, object Value)> _selectors = new List<(MemberExpression Selector, object Value)>();

public ImmutableBuilder<T> With<TProperty>(Expression<Func<T, TProperty>> selector, TProperty value)

_selectors.Add(((MemberExpression)selector.Body, value));
return this;


public T Build()

var ctors =
from ctor in typeof(T).GetConstructors()
let parameters = ctor.GetParameters()
// Join parameters and values by parameter order.
// The ctor requires them sorted but they might be initialized in any order.
let requiredParameterValues =
from parameter in parameters.Where(p => !p.IsOptional)
join selector in _selectors on (IgnoreCase)parameter.Name equals (IgnoreCase)selector.Selector.Member.Name
select selector.Value
// Get optional parameters if any.
let optionalParameterValues =
from parameter in parameters.Where(p => p.IsOptional)
join selector in _selectors on (IgnoreCase)parameter.Name equals (IgnoreCase)selector.Selector.Member.Name into s
from selector in s.DefaultIfEmpty()
select selector.Value
// Make sure all required parameters are specified.
where requiredParameterValues.Count() == parameters.Where(p => !p.IsOptional).Count()
select (ctor, requiredParameterValues, optionalParameterValues);

var theOne = ctors.Single();
return (T)theOne.ctor.Invoke(theOne.requiredParameterValues.Concat(theOne.optionalParameterValues).ToArray());



public static class Immutable<T>

public static ImmutableBuilder<T> Builder => new ImmutableBuilder<T>();



In case we already have an immutable type and want to modify it by changing just some values, the ImmutableHelper can be used. It also provides the With method expecting a new value. Then matches it and other properties with the constructor and uses origianl values as defaults and for the specified property the new value.



public static class ImmutableHelper

public static T With<T, TProperty>(this T obj, Expression<Func<T, TProperty>> selector, TProperty value)

var comparer = StringComparer.OrdinalIgnoreCase;

var propertyName = ((MemberExpression)selector.Body).Member.Name;
var properties = typeof(T).GetProperties();

var propertyNames =
properties
.Select(x => x.Name)
// A hash-set is convenient for name matching in the next step.
.ToImmutableHashSet(comparer);

// Find the constructor that matches property names.
var ctors =
from ctor in typeof(T).GetConstructors()
where propertyNames.IsProperSupersetOf(ctor.GetParameters().Select(x => x.Name))
select ctor;

var theOne = ctors.Single(); // There can be only one match.

var parameters =
from parameter in theOne.GetParameters()
join property in properties on (IgnoreCase)parameter.Name equals (IgnoreCase)property.Name
// They definitely match so no comparer is necessary.
// Use either the new value or the current one.
select property.Name == propertyName ? value : property.GetValue(obj);

return (T)theOne.Invoke(parameters.ToArray());




Example



This is how it can be used:



void Main()

var person =
Immutable<Person>
.Builder
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Build();

person.With(x => x.LastName, "Doe").Dump();


public class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";




What do you say? Crazy? Insane? I like it? Let's improve it?







c# generics reflection immutability expression-trees






share|improve this question















share|improve this question













share|improve this question




share|improve this question








edited 8 hours ago







t3chb0t

















asked 9 hours ago









t3chb0tt3chb0t

37.7k7 gold badges60 silver badges141 bronze badges




37.7k7 gold badges60 silver badges141 bronze badges










  • 2




    $begingroup$
    This has a use case for building controllers and what not that have dozens of parameters. Does this also work with optional ctr arguments?
    $endgroup$
    – dfhwze
    9 hours ago






  • 1




    $begingroup$
    @dfhwze it doesn't... but it definitelly could... and should.
    $endgroup$
    – t3chb0t
    9 hours ago










  • $begingroup$
    Ok ... so let's improve it! :p
    $endgroup$
    – dfhwze
    9 hours ago










  • $begingroup$
    @dfhwze done! ;-] at least the main builder can do this now.
    $endgroup$
    – t3chb0t
    8 hours ago







  • 1




    $begingroup$
    I like the With extensions method that helps to create modified copies of mutable types. However I didn't get the use case of the ImmutableBuilder.. Why not just calling the constructor?
    $endgroup$
    – JanDotNet
    5 hours ago












  • 2




    $begingroup$
    This has a use case for building controllers and what not that have dozens of parameters. Does this also work with optional ctr arguments?
    $endgroup$
    – dfhwze
    9 hours ago






  • 1




    $begingroup$
    @dfhwze it doesn't... but it definitelly could... and should.
    $endgroup$
    – t3chb0t
    9 hours ago










  • $begingroup$
    Ok ... so let's improve it! :p
    $endgroup$
    – dfhwze
    9 hours ago










  • $begingroup$
    @dfhwze done! ;-] at least the main builder can do this now.
    $endgroup$
    – t3chb0t
    8 hours ago







  • 1




    $begingroup$
    I like the With extensions method that helps to create modified copies of mutable types. However I didn't get the use case of the ImmutableBuilder.. Why not just calling the constructor?
    $endgroup$
    – JanDotNet
    5 hours ago







2




2




$begingroup$
This has a use case for building controllers and what not that have dozens of parameters. Does this also work with optional ctr arguments?
$endgroup$
– dfhwze
9 hours ago




$begingroup$
This has a use case for building controllers and what not that have dozens of parameters. Does this also work with optional ctr arguments?
$endgroup$
– dfhwze
9 hours ago




1




1




$begingroup$
@dfhwze it doesn't... but it definitelly could... and should.
$endgroup$
– t3chb0t
9 hours ago




$begingroup$
@dfhwze it doesn't... but it definitelly could... and should.
$endgroup$
– t3chb0t
9 hours ago












$begingroup$
Ok ... so let's improve it! :p
$endgroup$
– dfhwze
9 hours ago




$begingroup$
Ok ... so let's improve it! :p
$endgroup$
– dfhwze
9 hours ago












$begingroup$
@dfhwze done! ;-] at least the main builder can do this now.
$endgroup$
– t3chb0t
8 hours ago





$begingroup$
@dfhwze done! ;-] at least the main builder can do this now.
$endgroup$
– t3chb0t
8 hours ago





1




1




$begingroup$
I like the With extensions method that helps to create modified copies of mutable types. However I didn't get the use case of the ImmutableBuilder.. Why not just calling the constructor?
$endgroup$
– JanDotNet
5 hours ago




$begingroup$
I like the With extensions method that helps to create modified copies of mutable types. However I didn't get the use case of the ImmutableBuilder.. Why not just calling the constructor?
$endgroup$
– JanDotNet
5 hours ago










2 Answers
2






active

oldest

votes


















7













$begingroup$

You use IList<> where you should use ICollection<>. I've rarely encountered a scenario where IList<> actually needs to be used. The ICollection<> interface has most of the list's methods, but without everything related to indexing, which you don't use anyway. It's not that big of a deal, but I think it's good knowledge.




When you search for constructor parameters, I think you should match on parameter types in addition of parameter names. The reason behind this is that this that parameter types with names are guaranteed to be unique, where parameter names could not. For example (it's not a great one, but it's a case that would make your code potentially crash)



class Foobar


public Foobar(string a, int b)



public Foobar(string a, double b)







One problem I see with the Immutable<T> class is that I wouldn't expect a static property to return a new reference every time I call it. I've been trying to find another example in the .NET framework of when this happens, and... I couldn't. I would change it for a method named something like CreateBuilder() or something like that, this way it's clear that we are using a new builder every time we call the method.




I think the Immutable<> type is misleading. When seeing this, I'd expect being able to use it to make some mutable type immutable (however that would be done), but that's not what it does. As a matter of fact, most of your code doesn't rely on the T type to be immutable, which makes me think a tweak or two could make your tool work on mutable types too. In that sense, claiming it's an ImmutableBuilder is kind of wrong, it's a builder that works on immutable types, but on mutable types too.




According to comments, your With method in ImmutableHelper creates a copy of the object with a changed parameter, which is alright considering it doesn't modify the immutable type. What I think could be improved is a similar method with the signature static T With<T, TProperty>(this T obj, IEnmerable<(Expression<Func<T, TProperty>> selector, TProperty value)>), so that if you want to modify more than one field in the object you could do so without having to create a copy of the object every time.






share|improve this answer











$endgroup$














  • $begingroup$
    About the last paragraph... this will go into production (after I made it match parameter types too) because let's say you have an immutable type Uri with such a constructor as scheme, authority, path, query, fragment and you just want to change its scheme from nothing (relative uri) to http and maybe add a fragment. It's mutch easier and reliable to do this with With(x => x.Property, value) and let the framework handle other values than copy all the values yourself... and I once made that mistake of forgetting one of them... I was chasing the bug for 3h :-[
    $endgroup$
    – t3chb0t
    7 hours ago











  • $begingroup$
    @t3chb0t But by doing this you break everything that is expected from an immutable type. If another developer uses the Uri class, he/she will expect it to be immutable. If you happen to use your method somewhere in the code, debugging for this other person will be much more hellish than otherwise. You could use a Copy method then change the necessary fields or something else, maybe.
    $endgroup$
    – IEatBagels
    6 hours ago






  • 1




    $begingroup$
    @t3chb0t Although I understand the bug you've had, it's a tricky bug to find because it's kind of unexpected. I just don't think mutating immutable types is the right solution. Otherwise, maybe your With method could create a new copy of the object
    $endgroup$
    – IEatBagels
    6 hours ago










  • $begingroup$
    Oh, you mean this... yes! It is creating a new copy. The first class is just an initializer and the extension ImmutableHelper.With copies values into a new object. I probably should have put quotes around updater.
    $endgroup$
    – t3chb0t
    6 hours ago











  • $begingroup$
    @t3chb0t oohhh. Let me edit my answer then.
    $endgroup$
    – IEatBagels
    6 hours ago



















0













$begingroup$

(self-answer)




The idea for not-using the constructor is indeed insane but... since it's possible I kept it in the new version too. I changed the name of this tool to DtoUpdater. It now can collect updates for multiple properties that at the end have to be Commited. Parameters are now matched with members not only by name but also by type and it picks the constructor with the most parameters ana matching properties. I created this helper for updating simple DTOs and they usually have only one constructor initializing all properties so I think its current complexity is sufficient for most use-cases.



This version also no longer forces the user to specify all values that a constructor requires. I think that this new functionality now makes this utility in certain situations more useful than using the construtor... if it allows default values of course.



public static class DtoUpdater

public static DtoUpdater<T> For<T>() => new DtoUpdater<T>(default);

public static DtoUpdater<T> Update<T>(this T obj) => new DtoUpdater<T>(obj);


public class DtoUpdater<T>

private readonly T _obj;

private readonly ICollection<(MemberInfo Member, object Value)> _updates = new List<(MemberInfo Member, object Value)>();

public DtoUpdater(T obj) => _obj = obj;

public DtoUpdater<T> With<TProperty>(Expression<Func<T, TProperty>> update, TProperty value)

_updates.Add((((MemberExpression)update.Body).Member, value));
return this;


public T Commit()

var members =
from member in typeof(T).GetMembers(BindingFlags.Public

private object GetMemberValueOrDefault(string memberName)

if (_obj == null) return default;

// There is for sure only one member with that name.
switch (typeof(T).GetMembers(BindingFlags.Public


public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




I hid the IgnoreCase helper behind a new extension:



public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




The new API can now be used like this:



public class DtoBuilderTest

[Fact]
public void Can_create_and_update_object()

var person =
DtoUpdater
.For<Person>()
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Null(person.LastName);
Assert.Null(person.NickName);

person =
person
.Update()
.With(x => x.LastName, "Doe")
.With(x => x.NickName, "JD")
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Equal("Doe", person.LastName);
Assert.Equal("JD", person.NickName);


private class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get; set;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";







share|improve this answer









$endgroup$










  • 1




    $begingroup$
    This update/commit pattern makes me think we can tune this up to be used as a batch updater with support for multi-level change tracking/reverting implementing IEditableObject, IChangeTracking and IRevertibleChangeTracking (up-vote pending btw)
    $endgroup$
    – dfhwze
    3 hours ago











  • $begingroup$
    @dfhwze haha don't even think of it. Two more refactoring rounds and it can fly to the moon ;-P
    $endgroup$
    – t3chb0t
    3 hours ago










  • $begingroup$
    I might ask a follow-up question about it in the weekend o_0 But change tracking is not that trivial /o
    $endgroup$
    – dfhwze
    3 hours ago










  • $begingroup$
    @dfhwze I'm not so sure we need all this tracking and undo logic. This demo doesn't show that but usually when you modify an immutable, you pass it so some deeper scope. When it returns, you pick up the previous version for this scope and continue from here. At leat this is how I mostly use immutables. They are convenient to update and use in a new context for a moment. Although it might be useful for debugging.
    $endgroup$
    – t3chb0t
    3 hours ago











  • $begingroup$
    We don't need it for this use case, I want to stea.. 'borrow' your code to make a change tracking API :p Basically, I want your immutable code for my mutable stuff. That's it, I'm no longer making any sense at all ... I'm off to bed, see ya.
    $endgroup$
    – dfhwze
    3 hours ago














Your Answer






StackExchange.ifUsing("editor", function ()
StackExchange.using("externalEditor", function ()
StackExchange.using("snippets", function ()
StackExchange.snippets.init();
);
);
, "code-snippets");

StackExchange.ready(function()
var channelOptions =
tags: "".split(" "),
id: "196"
;
initTagRenderer("".split(" "), "".split(" "), channelOptions);

StackExchange.using("externalEditor", function()
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled)
StackExchange.using("snippets", function()
createEditor();
);

else
createEditor();

);

function createEditor()
StackExchange.prepareEditor(
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: false,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: null,
bindNavPrevention: true,
postfix: "",
imageUploader:
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
,
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
);



);













draft saved

draft discarded


















StackExchange.ready(
function ()
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f226694%2fimmutable-builder-and-updater%23new-answer', 'question_page');

);

Post as a guest















Required, but never shown

























2 Answers
2






active

oldest

votes








2 Answers
2






active

oldest

votes









active

oldest

votes






active

oldest

votes









7













$begingroup$

You use IList<> where you should use ICollection<>. I've rarely encountered a scenario where IList<> actually needs to be used. The ICollection<> interface has most of the list's methods, but without everything related to indexing, which you don't use anyway. It's not that big of a deal, but I think it's good knowledge.




When you search for constructor parameters, I think you should match on parameter types in addition of parameter names. The reason behind this is that this that parameter types with names are guaranteed to be unique, where parameter names could not. For example (it's not a great one, but it's a case that would make your code potentially crash)



class Foobar


public Foobar(string a, int b)



public Foobar(string a, double b)







One problem I see with the Immutable<T> class is that I wouldn't expect a static property to return a new reference every time I call it. I've been trying to find another example in the .NET framework of when this happens, and... I couldn't. I would change it for a method named something like CreateBuilder() or something like that, this way it's clear that we are using a new builder every time we call the method.




I think the Immutable<> type is misleading. When seeing this, I'd expect being able to use it to make some mutable type immutable (however that would be done), but that's not what it does. As a matter of fact, most of your code doesn't rely on the T type to be immutable, which makes me think a tweak or two could make your tool work on mutable types too. In that sense, claiming it's an ImmutableBuilder is kind of wrong, it's a builder that works on immutable types, but on mutable types too.




According to comments, your With method in ImmutableHelper creates a copy of the object with a changed parameter, which is alright considering it doesn't modify the immutable type. What I think could be improved is a similar method with the signature static T With<T, TProperty>(this T obj, IEnmerable<(Expression<Func<T, TProperty>> selector, TProperty value)>), so that if you want to modify more than one field in the object you could do so without having to create a copy of the object every time.






share|improve this answer











$endgroup$














  • $begingroup$
    About the last paragraph... this will go into production (after I made it match parameter types too) because let's say you have an immutable type Uri with such a constructor as scheme, authority, path, query, fragment and you just want to change its scheme from nothing (relative uri) to http and maybe add a fragment. It's mutch easier and reliable to do this with With(x => x.Property, value) and let the framework handle other values than copy all the values yourself... and I once made that mistake of forgetting one of them... I was chasing the bug for 3h :-[
    $endgroup$
    – t3chb0t
    7 hours ago











  • $begingroup$
    @t3chb0t But by doing this you break everything that is expected from an immutable type. If another developer uses the Uri class, he/she will expect it to be immutable. If you happen to use your method somewhere in the code, debugging for this other person will be much more hellish than otherwise. You could use a Copy method then change the necessary fields or something else, maybe.
    $endgroup$
    – IEatBagels
    6 hours ago






  • 1




    $begingroup$
    @t3chb0t Although I understand the bug you've had, it's a tricky bug to find because it's kind of unexpected. I just don't think mutating immutable types is the right solution. Otherwise, maybe your With method could create a new copy of the object
    $endgroup$
    – IEatBagels
    6 hours ago










  • $begingroup$
    Oh, you mean this... yes! It is creating a new copy. The first class is just an initializer and the extension ImmutableHelper.With copies values into a new object. I probably should have put quotes around updater.
    $endgroup$
    – t3chb0t
    6 hours ago











  • $begingroup$
    @t3chb0t oohhh. Let me edit my answer then.
    $endgroup$
    – IEatBagels
    6 hours ago
















7













$begingroup$

You use IList<> where you should use ICollection<>. I've rarely encountered a scenario where IList<> actually needs to be used. The ICollection<> interface has most of the list's methods, but without everything related to indexing, which you don't use anyway. It's not that big of a deal, but I think it's good knowledge.




When you search for constructor parameters, I think you should match on parameter types in addition of parameter names. The reason behind this is that this that parameter types with names are guaranteed to be unique, where parameter names could not. For example (it's not a great one, but it's a case that would make your code potentially crash)



class Foobar


public Foobar(string a, int b)



public Foobar(string a, double b)







One problem I see with the Immutable<T> class is that I wouldn't expect a static property to return a new reference every time I call it. I've been trying to find another example in the .NET framework of when this happens, and... I couldn't. I would change it for a method named something like CreateBuilder() or something like that, this way it's clear that we are using a new builder every time we call the method.




I think the Immutable<> type is misleading. When seeing this, I'd expect being able to use it to make some mutable type immutable (however that would be done), but that's not what it does. As a matter of fact, most of your code doesn't rely on the T type to be immutable, which makes me think a tweak or two could make your tool work on mutable types too. In that sense, claiming it's an ImmutableBuilder is kind of wrong, it's a builder that works on immutable types, but on mutable types too.




According to comments, your With method in ImmutableHelper creates a copy of the object with a changed parameter, which is alright considering it doesn't modify the immutable type. What I think could be improved is a similar method with the signature static T With<T, TProperty>(this T obj, IEnmerable<(Expression<Func<T, TProperty>> selector, TProperty value)>), so that if you want to modify more than one field in the object you could do so without having to create a copy of the object every time.






share|improve this answer











$endgroup$














  • $begingroup$
    About the last paragraph... this will go into production (after I made it match parameter types too) because let's say you have an immutable type Uri with such a constructor as scheme, authority, path, query, fragment and you just want to change its scheme from nothing (relative uri) to http and maybe add a fragment. It's mutch easier and reliable to do this with With(x => x.Property, value) and let the framework handle other values than copy all the values yourself... and I once made that mistake of forgetting one of them... I was chasing the bug for 3h :-[
    $endgroup$
    – t3chb0t
    7 hours ago











  • $begingroup$
    @t3chb0t But by doing this you break everything that is expected from an immutable type. If another developer uses the Uri class, he/she will expect it to be immutable. If you happen to use your method somewhere in the code, debugging for this other person will be much more hellish than otherwise. You could use a Copy method then change the necessary fields or something else, maybe.
    $endgroup$
    – IEatBagels
    6 hours ago






  • 1




    $begingroup$
    @t3chb0t Although I understand the bug you've had, it's a tricky bug to find because it's kind of unexpected. I just don't think mutating immutable types is the right solution. Otherwise, maybe your With method could create a new copy of the object
    $endgroup$
    – IEatBagels
    6 hours ago










  • $begingroup$
    Oh, you mean this... yes! It is creating a new copy. The first class is just an initializer and the extension ImmutableHelper.With copies values into a new object. I probably should have put quotes around updater.
    $endgroup$
    – t3chb0t
    6 hours ago











  • $begingroup$
    @t3chb0t oohhh. Let me edit my answer then.
    $endgroup$
    – IEatBagels
    6 hours ago














7














7










7







$begingroup$

You use IList<> where you should use ICollection<>. I've rarely encountered a scenario where IList<> actually needs to be used. The ICollection<> interface has most of the list's methods, but without everything related to indexing, which you don't use anyway. It's not that big of a deal, but I think it's good knowledge.




When you search for constructor parameters, I think you should match on parameter types in addition of parameter names. The reason behind this is that this that parameter types with names are guaranteed to be unique, where parameter names could not. For example (it's not a great one, but it's a case that would make your code potentially crash)



class Foobar


public Foobar(string a, int b)



public Foobar(string a, double b)







One problem I see with the Immutable<T> class is that I wouldn't expect a static property to return a new reference every time I call it. I've been trying to find another example in the .NET framework of when this happens, and... I couldn't. I would change it for a method named something like CreateBuilder() or something like that, this way it's clear that we are using a new builder every time we call the method.




I think the Immutable<> type is misleading. When seeing this, I'd expect being able to use it to make some mutable type immutable (however that would be done), but that's not what it does. As a matter of fact, most of your code doesn't rely on the T type to be immutable, which makes me think a tweak or two could make your tool work on mutable types too. In that sense, claiming it's an ImmutableBuilder is kind of wrong, it's a builder that works on immutable types, but on mutable types too.




According to comments, your With method in ImmutableHelper creates a copy of the object with a changed parameter, which is alright considering it doesn't modify the immutable type. What I think could be improved is a similar method with the signature static T With<T, TProperty>(this T obj, IEnmerable<(Expression<Func<T, TProperty>> selector, TProperty value)>), so that if you want to modify more than one field in the object you could do so without having to create a copy of the object every time.






share|improve this answer











$endgroup$



You use IList<> where you should use ICollection<>. I've rarely encountered a scenario where IList<> actually needs to be used. The ICollection<> interface has most of the list's methods, but without everything related to indexing, which you don't use anyway. It's not that big of a deal, but I think it's good knowledge.




When you search for constructor parameters, I think you should match on parameter types in addition of parameter names. The reason behind this is that this that parameter types with names are guaranteed to be unique, where parameter names could not. For example (it's not a great one, but it's a case that would make your code potentially crash)



class Foobar


public Foobar(string a, int b)



public Foobar(string a, double b)







One problem I see with the Immutable<T> class is that I wouldn't expect a static property to return a new reference every time I call it. I've been trying to find another example in the .NET framework of when this happens, and... I couldn't. I would change it for a method named something like CreateBuilder() or something like that, this way it's clear that we are using a new builder every time we call the method.




I think the Immutable<> type is misleading. When seeing this, I'd expect being able to use it to make some mutable type immutable (however that would be done), but that's not what it does. As a matter of fact, most of your code doesn't rely on the T type to be immutable, which makes me think a tweak or two could make your tool work on mutable types too. In that sense, claiming it's an ImmutableBuilder is kind of wrong, it's a builder that works on immutable types, but on mutable types too.




According to comments, your With method in ImmutableHelper creates a copy of the object with a changed parameter, which is alright considering it doesn't modify the immutable type. What I think could be improved is a similar method with the signature static T With<T, TProperty>(this T obj, IEnmerable<(Expression<Func<T, TProperty>> selector, TProperty value)>), so that if you want to modify more than one field in the object you could do so without having to create a copy of the object every time.







share|improve this answer














share|improve this answer



share|improve this answer








edited 6 hours ago

























answered 7 hours ago









IEatBagelsIEatBagels

9,9502 gold badges35 silver badges86 bronze badges




9,9502 gold badges35 silver badges86 bronze badges














  • $begingroup$
    About the last paragraph... this will go into production (after I made it match parameter types too) because let's say you have an immutable type Uri with such a constructor as scheme, authority, path, query, fragment and you just want to change its scheme from nothing (relative uri) to http and maybe add a fragment. It's mutch easier and reliable to do this with With(x => x.Property, value) and let the framework handle other values than copy all the values yourself... and I once made that mistake of forgetting one of them... I was chasing the bug for 3h :-[
    $endgroup$
    – t3chb0t
    7 hours ago











  • $begingroup$
    @t3chb0t But by doing this you break everything that is expected from an immutable type. If another developer uses the Uri class, he/she will expect it to be immutable. If you happen to use your method somewhere in the code, debugging for this other person will be much more hellish than otherwise. You could use a Copy method then change the necessary fields or something else, maybe.
    $endgroup$
    – IEatBagels
    6 hours ago






  • 1




    $begingroup$
    @t3chb0t Although I understand the bug you've had, it's a tricky bug to find because it's kind of unexpected. I just don't think mutating immutable types is the right solution. Otherwise, maybe your With method could create a new copy of the object
    $endgroup$
    – IEatBagels
    6 hours ago










  • $begingroup$
    Oh, you mean this... yes! It is creating a new copy. The first class is just an initializer and the extension ImmutableHelper.With copies values into a new object. I probably should have put quotes around updater.
    $endgroup$
    – t3chb0t
    6 hours ago











  • $begingroup$
    @t3chb0t oohhh. Let me edit my answer then.
    $endgroup$
    – IEatBagels
    6 hours ago

















  • $begingroup$
    About the last paragraph... this will go into production (after I made it match parameter types too) because let's say you have an immutable type Uri with such a constructor as scheme, authority, path, query, fragment and you just want to change its scheme from nothing (relative uri) to http and maybe add a fragment. It's mutch easier and reliable to do this with With(x => x.Property, value) and let the framework handle other values than copy all the values yourself... and I once made that mistake of forgetting one of them... I was chasing the bug for 3h :-[
    $endgroup$
    – t3chb0t
    7 hours ago











  • $begingroup$
    @t3chb0t But by doing this you break everything that is expected from an immutable type. If another developer uses the Uri class, he/she will expect it to be immutable. If you happen to use your method somewhere in the code, debugging for this other person will be much more hellish than otherwise. You could use a Copy method then change the necessary fields or something else, maybe.
    $endgroup$
    – IEatBagels
    6 hours ago






  • 1




    $begingroup$
    @t3chb0t Although I understand the bug you've had, it's a tricky bug to find because it's kind of unexpected. I just don't think mutating immutable types is the right solution. Otherwise, maybe your With method could create a new copy of the object
    $endgroup$
    – IEatBagels
    6 hours ago










  • $begingroup$
    Oh, you mean this... yes! It is creating a new copy. The first class is just an initializer and the extension ImmutableHelper.With copies values into a new object. I probably should have put quotes around updater.
    $endgroup$
    – t3chb0t
    6 hours ago











  • $begingroup$
    @t3chb0t oohhh. Let me edit my answer then.
    $endgroup$
    – IEatBagels
    6 hours ago
















$begingroup$
About the last paragraph... this will go into production (after I made it match parameter types too) because let's say you have an immutable type Uri with such a constructor as scheme, authority, path, query, fragment and you just want to change its scheme from nothing (relative uri) to http and maybe add a fragment. It's mutch easier and reliable to do this with With(x => x.Property, value) and let the framework handle other values than copy all the values yourself... and I once made that mistake of forgetting one of them... I was chasing the bug for 3h :-[
$endgroup$
– t3chb0t
7 hours ago





$begingroup$
About the last paragraph... this will go into production (after I made it match parameter types too) because let's say you have an immutable type Uri with such a constructor as scheme, authority, path, query, fragment and you just want to change its scheme from nothing (relative uri) to http and maybe add a fragment. It's mutch easier and reliable to do this with With(x => x.Property, value) and let the framework handle other values than copy all the values yourself... and I once made that mistake of forgetting one of them... I was chasing the bug for 3h :-[
$endgroup$
– t3chb0t
7 hours ago













$begingroup$
@t3chb0t But by doing this you break everything that is expected from an immutable type. If another developer uses the Uri class, he/she will expect it to be immutable. If you happen to use your method somewhere in the code, debugging for this other person will be much more hellish than otherwise. You could use a Copy method then change the necessary fields or something else, maybe.
$endgroup$
– IEatBagels
6 hours ago




$begingroup$
@t3chb0t But by doing this you break everything that is expected from an immutable type. If another developer uses the Uri class, he/she will expect it to be immutable. If you happen to use your method somewhere in the code, debugging for this other person will be much more hellish than otherwise. You could use a Copy method then change the necessary fields or something else, maybe.
$endgroup$
– IEatBagels
6 hours ago




1




1




$begingroup$
@t3chb0t Although I understand the bug you've had, it's a tricky bug to find because it's kind of unexpected. I just don't think mutating immutable types is the right solution. Otherwise, maybe your With method could create a new copy of the object
$endgroup$
– IEatBagels
6 hours ago




$begingroup$
@t3chb0t Although I understand the bug you've had, it's a tricky bug to find because it's kind of unexpected. I just don't think mutating immutable types is the right solution. Otherwise, maybe your With method could create a new copy of the object
$endgroup$
– IEatBagels
6 hours ago












$begingroup$
Oh, you mean this... yes! It is creating a new copy. The first class is just an initializer and the extension ImmutableHelper.With copies values into a new object. I probably should have put quotes around updater.
$endgroup$
– t3chb0t
6 hours ago





$begingroup$
Oh, you mean this... yes! It is creating a new copy. The first class is just an initializer and the extension ImmutableHelper.With copies values into a new object. I probably should have put quotes around updater.
$endgroup$
– t3chb0t
6 hours ago













$begingroup$
@t3chb0t oohhh. Let me edit my answer then.
$endgroup$
– IEatBagels
6 hours ago





$begingroup$
@t3chb0t oohhh. Let me edit my answer then.
$endgroup$
– IEatBagels
6 hours ago














0













$begingroup$

(self-answer)




The idea for not-using the constructor is indeed insane but... since it's possible I kept it in the new version too. I changed the name of this tool to DtoUpdater. It now can collect updates for multiple properties that at the end have to be Commited. Parameters are now matched with members not only by name but also by type and it picks the constructor with the most parameters ana matching properties. I created this helper for updating simple DTOs and they usually have only one constructor initializing all properties so I think its current complexity is sufficient for most use-cases.



This version also no longer forces the user to specify all values that a constructor requires. I think that this new functionality now makes this utility in certain situations more useful than using the construtor... if it allows default values of course.



public static class DtoUpdater

public static DtoUpdater<T> For<T>() => new DtoUpdater<T>(default);

public static DtoUpdater<T> Update<T>(this T obj) => new DtoUpdater<T>(obj);


public class DtoUpdater<T>

private readonly T _obj;

private readonly ICollection<(MemberInfo Member, object Value)> _updates = new List<(MemberInfo Member, object Value)>();

public DtoUpdater(T obj) => _obj = obj;

public DtoUpdater<T> With<TProperty>(Expression<Func<T, TProperty>> update, TProperty value)

_updates.Add((((MemberExpression)update.Body).Member, value));
return this;


public T Commit()

var members =
from member in typeof(T).GetMembers(BindingFlags.Public

private object GetMemberValueOrDefault(string memberName)

if (_obj == null) return default;

// There is for sure only one member with that name.
switch (typeof(T).GetMembers(BindingFlags.Public


public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




I hid the IgnoreCase helper behind a new extension:



public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




The new API can now be used like this:



public class DtoBuilderTest

[Fact]
public void Can_create_and_update_object()

var person =
DtoUpdater
.For<Person>()
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Null(person.LastName);
Assert.Null(person.NickName);

person =
person
.Update()
.With(x => x.LastName, "Doe")
.With(x => x.NickName, "JD")
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Equal("Doe", person.LastName);
Assert.Equal("JD", person.NickName);


private class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get; set;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";







share|improve this answer









$endgroup$










  • 1




    $begingroup$
    This update/commit pattern makes me think we can tune this up to be used as a batch updater with support for multi-level change tracking/reverting implementing IEditableObject, IChangeTracking and IRevertibleChangeTracking (up-vote pending btw)
    $endgroup$
    – dfhwze
    3 hours ago











  • $begingroup$
    @dfhwze haha don't even think of it. Two more refactoring rounds and it can fly to the moon ;-P
    $endgroup$
    – t3chb0t
    3 hours ago










  • $begingroup$
    I might ask a follow-up question about it in the weekend o_0 But change tracking is not that trivial /o
    $endgroup$
    – dfhwze
    3 hours ago










  • $begingroup$
    @dfhwze I'm not so sure we need all this tracking and undo logic. This demo doesn't show that but usually when you modify an immutable, you pass it so some deeper scope. When it returns, you pick up the previous version for this scope and continue from here. At leat this is how I mostly use immutables. They are convenient to update and use in a new context for a moment. Although it might be useful for debugging.
    $endgroup$
    – t3chb0t
    3 hours ago











  • $begingroup$
    We don't need it for this use case, I want to stea.. 'borrow' your code to make a change tracking API :p Basically, I want your immutable code for my mutable stuff. That's it, I'm no longer making any sense at all ... I'm off to bed, see ya.
    $endgroup$
    – dfhwze
    3 hours ago
















0













$begingroup$

(self-answer)




The idea for not-using the constructor is indeed insane but... since it's possible I kept it in the new version too. I changed the name of this tool to DtoUpdater. It now can collect updates for multiple properties that at the end have to be Commited. Parameters are now matched with members not only by name but also by type and it picks the constructor with the most parameters ana matching properties. I created this helper for updating simple DTOs and they usually have only one constructor initializing all properties so I think its current complexity is sufficient for most use-cases.



This version also no longer forces the user to specify all values that a constructor requires. I think that this new functionality now makes this utility in certain situations more useful than using the construtor... if it allows default values of course.



public static class DtoUpdater

public static DtoUpdater<T> For<T>() => new DtoUpdater<T>(default);

public static DtoUpdater<T> Update<T>(this T obj) => new DtoUpdater<T>(obj);


public class DtoUpdater<T>

private readonly T _obj;

private readonly ICollection<(MemberInfo Member, object Value)> _updates = new List<(MemberInfo Member, object Value)>();

public DtoUpdater(T obj) => _obj = obj;

public DtoUpdater<T> With<TProperty>(Expression<Func<T, TProperty>> update, TProperty value)

_updates.Add((((MemberExpression)update.Body).Member, value));
return this;


public T Commit()

var members =
from member in typeof(T).GetMembers(BindingFlags.Public

private object GetMemberValueOrDefault(string memberName)

if (_obj == null) return default;

// There is for sure only one member with that name.
switch (typeof(T).GetMembers(BindingFlags.Public


public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




I hid the IgnoreCase helper behind a new extension:



public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




The new API can now be used like this:



public class DtoBuilderTest

[Fact]
public void Can_create_and_update_object()

var person =
DtoUpdater
.For<Person>()
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Null(person.LastName);
Assert.Null(person.NickName);

person =
person
.Update()
.With(x => x.LastName, "Doe")
.With(x => x.NickName, "JD")
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Equal("Doe", person.LastName);
Assert.Equal("JD", person.NickName);


private class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get; set;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";







share|improve this answer









$endgroup$










  • 1




    $begingroup$
    This update/commit pattern makes me think we can tune this up to be used as a batch updater with support for multi-level change tracking/reverting implementing IEditableObject, IChangeTracking and IRevertibleChangeTracking (up-vote pending btw)
    $endgroup$
    – dfhwze
    3 hours ago











  • $begingroup$
    @dfhwze haha don't even think of it. Two more refactoring rounds and it can fly to the moon ;-P
    $endgroup$
    – t3chb0t
    3 hours ago










  • $begingroup$
    I might ask a follow-up question about it in the weekend o_0 But change tracking is not that trivial /o
    $endgroup$
    – dfhwze
    3 hours ago










  • $begingroup$
    @dfhwze I'm not so sure we need all this tracking and undo logic. This demo doesn't show that but usually when you modify an immutable, you pass it so some deeper scope. When it returns, you pick up the previous version for this scope and continue from here. At leat this is how I mostly use immutables. They are convenient to update and use in a new context for a moment. Although it might be useful for debugging.
    $endgroup$
    – t3chb0t
    3 hours ago











  • $begingroup$
    We don't need it for this use case, I want to stea.. 'borrow' your code to make a change tracking API :p Basically, I want your immutable code for my mutable stuff. That's it, I'm no longer making any sense at all ... I'm off to bed, see ya.
    $endgroup$
    – dfhwze
    3 hours ago














0














0










0







$begingroup$

(self-answer)




The idea for not-using the constructor is indeed insane but... since it's possible I kept it in the new version too. I changed the name of this tool to DtoUpdater. It now can collect updates for multiple properties that at the end have to be Commited. Parameters are now matched with members not only by name but also by type and it picks the constructor with the most parameters ana matching properties. I created this helper for updating simple DTOs and they usually have only one constructor initializing all properties so I think its current complexity is sufficient for most use-cases.



This version also no longer forces the user to specify all values that a constructor requires. I think that this new functionality now makes this utility in certain situations more useful than using the construtor... if it allows default values of course.



public static class DtoUpdater

public static DtoUpdater<T> For<T>() => new DtoUpdater<T>(default);

public static DtoUpdater<T> Update<T>(this T obj) => new DtoUpdater<T>(obj);


public class DtoUpdater<T>

private readonly T _obj;

private readonly ICollection<(MemberInfo Member, object Value)> _updates = new List<(MemberInfo Member, object Value)>();

public DtoUpdater(T obj) => _obj = obj;

public DtoUpdater<T> With<TProperty>(Expression<Func<T, TProperty>> update, TProperty value)

_updates.Add((((MemberExpression)update.Body).Member, value));
return this;


public T Commit()

var members =
from member in typeof(T).GetMembers(BindingFlags.Public

private object GetMemberValueOrDefault(string memberName)

if (_obj == null) return default;

// There is for sure only one member with that name.
switch (typeof(T).GetMembers(BindingFlags.Public


public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




I hid the IgnoreCase helper behind a new extension:



public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




The new API can now be used like this:



public class DtoBuilderTest

[Fact]
public void Can_create_and_update_object()

var person =
DtoUpdater
.For<Person>()
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Null(person.LastName);
Assert.Null(person.NickName);

person =
person
.Update()
.With(x => x.LastName, "Doe")
.With(x => x.NickName, "JD")
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Equal("Doe", person.LastName);
Assert.Equal("JD", person.NickName);


private class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get; set;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";







share|improve this answer









$endgroup$



(self-answer)




The idea for not-using the constructor is indeed insane but... since it's possible I kept it in the new version too. I changed the name of this tool to DtoUpdater. It now can collect updates for multiple properties that at the end have to be Commited. Parameters are now matched with members not only by name but also by type and it picks the constructor with the most parameters ana matching properties. I created this helper for updating simple DTOs and they usually have only one constructor initializing all properties so I think its current complexity is sufficient for most use-cases.



This version also no longer forces the user to specify all values that a constructor requires. I think that this new functionality now makes this utility in certain situations more useful than using the construtor... if it allows default values of course.



public static class DtoUpdater

public static DtoUpdater<T> For<T>() => new DtoUpdater<T>(default);

public static DtoUpdater<T> Update<T>(this T obj) => new DtoUpdater<T>(obj);


public class DtoUpdater<T>

private readonly T _obj;

private readonly ICollection<(MemberInfo Member, object Value)> _updates = new List<(MemberInfo Member, object Value)>();

public DtoUpdater(T obj) => _obj = obj;

public DtoUpdater<T> With<TProperty>(Expression<Func<T, TProperty>> update, TProperty value)

_updates.Add((((MemberExpression)update.Body).Member, value));
return this;


public T Commit()

var members =
from member in typeof(T).GetMembers(BindingFlags.Public

private object GetMemberValueOrDefault(string memberName)

if (_obj == null) return default;

// There is for sure only one member with that name.
switch (typeof(T).GetMembers(BindingFlags.Public


public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




I hid the IgnoreCase helper behind a new extension:



public static class StringExtensions

public static IEquatable<string> AsIgnoreCase(this string str) => (IgnoreCase)str;

private class IgnoreCase : IEquatable<string>

private IgnoreCase(string value) => Value = value;
private string Value get;
public bool Equals(string other) => StringComparer.OrdinalIgnoreCase.Equals(Value, other);
public override bool Equals(object obj) => obj is IgnoreCase ic && Equals(ic.Value);
public override int GetHashCode() => StringComparer.OrdinalIgnoreCase.GetHashCode(Value);
public static explicit operator IgnoreCase(string value) => new IgnoreCase(value);




The new API can now be used like this:



public class DtoBuilderTest

[Fact]
public void Can_create_and_update_object()

var person =
DtoUpdater
.For<Person>()
.With(x => x.FirstName, "Jane")
.With(x => x.LastName, null)
//.With(x => x.NickName, "JD") // Optional
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Null(person.LastName);
Assert.Null(person.NickName);

person =
person
.Update()
.With(x => x.LastName, "Doe")
.With(x => x.NickName, "JD")
.Commit();

Assert.Equal("Jane", person.FirstName);
Assert.Equal("Doe", person.LastName);
Assert.Equal("JD", person.NickName);


private class Person

public Person(string firstName, string lastName, string nickName = null)

FirstName = firstName;
LastName = lastName;
NickName = nickName;


// This ctor should confuse the API.
public Person(string other)

public string FirstName get;

public string LastName get;

public string NickName get; set;

// This property should confuse the API too.
public string FullName => $"LastName, FirstName";








share|improve this answer












share|improve this answer



share|improve this answer










answered 4 hours ago









t3chb0tt3chb0t

37.7k7 gold badges60 silver badges141 bronze badges




37.7k7 gold badges60 silver badges141 bronze badges










  • 1




    $begingroup$
    This update/commit pattern makes me think we can tune this up to be used as a batch updater with support for multi-level change tracking/reverting implementing IEditableObject, IChangeTracking and IRevertibleChangeTracking (up-vote pending btw)
    $endgroup$
    – dfhwze
    3 hours ago











  • $begingroup$
    @dfhwze haha don't even think of it. Two more refactoring rounds and it can fly to the moon ;-P
    $endgroup$
    – t3chb0t
    3 hours ago










  • $begingroup$
    I might ask a follow-up question about it in the weekend o_0 But change tracking is not that trivial /o
    $endgroup$
    – dfhwze
    3 hours ago










  • $begingroup$
    @dfhwze I'm not so sure we need all this tracking and undo logic. This demo doesn't show that but usually when you modify an immutable, you pass it so some deeper scope. When it returns, you pick up the previous version for this scope and continue from here. At leat this is how I mostly use immutables. They are convenient to update and use in a new context for a moment. Although it might be useful for debugging.
    $endgroup$
    – t3chb0t
    3 hours ago











  • $begingroup$
    We don't need it for this use case, I want to stea.. 'borrow' your code to make a change tracking API :p Basically, I want your immutable code for my mutable stuff. That's it, I'm no longer making any sense at all ... I'm off to bed, see ya.
    $endgroup$
    – dfhwze
    3 hours ago













  • 1




    $begingroup$
    This update/commit pattern makes me think we can tune this up to be used as a batch updater with support for multi-level change tracking/reverting implementing IEditableObject, IChangeTracking and IRevertibleChangeTracking (up-vote pending btw)
    $endgroup$
    – dfhwze
    3 hours ago











  • $begingroup$
    @dfhwze haha don't even think of it. Two more refactoring rounds and it can fly to the moon ;-P
    $endgroup$
    – t3chb0t
    3 hours ago










  • $begingroup$
    I might ask a follow-up question about it in the weekend o_0 But change tracking is not that trivial /o
    $endgroup$
    – dfhwze
    3 hours ago










  • $begingroup$
    @dfhwze I'm not so sure we need all this tracking and undo logic. This demo doesn't show that but usually when you modify an immutable, you pass it so some deeper scope. When it returns, you pick up the previous version for this scope and continue from here. At leat this is how I mostly use immutables. They are convenient to update and use in a new context for a moment. Although it might be useful for debugging.
    $endgroup$
    – t3chb0t
    3 hours ago











  • $begingroup$
    We don't need it for this use case, I want to stea.. 'borrow' your code to make a change tracking API :p Basically, I want your immutable code for my mutable stuff. That's it, I'm no longer making any sense at all ... I'm off to bed, see ya.
    $endgroup$
    – dfhwze
    3 hours ago








1




1




$begingroup$
This update/commit pattern makes me think we can tune this up to be used as a batch updater with support for multi-level change tracking/reverting implementing IEditableObject, IChangeTracking and IRevertibleChangeTracking (up-vote pending btw)
$endgroup$
– dfhwze
3 hours ago





$begingroup$
This update/commit pattern makes me think we can tune this up to be used as a batch updater with support for multi-level change tracking/reverting implementing IEditableObject, IChangeTracking and IRevertibleChangeTracking (up-vote pending btw)
$endgroup$
– dfhwze
3 hours ago













$begingroup$
@dfhwze haha don't even think of it. Two more refactoring rounds and it can fly to the moon ;-P
$endgroup$
– t3chb0t
3 hours ago




$begingroup$
@dfhwze haha don't even think of it. Two more refactoring rounds and it can fly to the moon ;-P
$endgroup$
– t3chb0t
3 hours ago












$begingroup$
I might ask a follow-up question about it in the weekend o_0 But change tracking is not that trivial /o
$endgroup$
– dfhwze
3 hours ago




$begingroup$
I might ask a follow-up question about it in the weekend o_0 But change tracking is not that trivial /o
$endgroup$
– dfhwze
3 hours ago












$begingroup$
@dfhwze I'm not so sure we need all this tracking and undo logic. This demo doesn't show that but usually when you modify an immutable, you pass it so some deeper scope. When it returns, you pick up the previous version for this scope and continue from here. At leat this is how I mostly use immutables. They are convenient to update and use in a new context for a moment. Although it might be useful for debugging.
$endgroup$
– t3chb0t
3 hours ago





$begingroup$
@dfhwze I'm not so sure we need all this tracking and undo logic. This demo doesn't show that but usually when you modify an immutable, you pass it so some deeper scope. When it returns, you pick up the previous version for this scope and continue from here. At leat this is how I mostly use immutables. They are convenient to update and use in a new context for a moment. Although it might be useful for debugging.
$endgroup$
– t3chb0t
3 hours ago













$begingroup$
We don't need it for this use case, I want to stea.. 'borrow' your code to make a change tracking API :p Basically, I want your immutable code for my mutable stuff. That's it, I'm no longer making any sense at all ... I'm off to bed, see ya.
$endgroup$
– dfhwze
3 hours ago





$begingroup$
We don't need it for this use case, I want to stea.. 'borrow' your code to make a change tracking API :p Basically, I want your immutable code for my mutable stuff. That's it, I'm no longer making any sense at all ... I'm off to bed, see ya.
$endgroup$
– dfhwze
3 hours ago


















draft saved

draft discarded
















































Thanks for contributing an answer to Code Review Stack Exchange!


  • Please be sure to answer the question. Provide details and share your research!

But avoid


  • Asking for help, clarification, or responding to other answers.

  • Making statements based on opinion; back them up with references or personal experience.

Use MathJax to format equations. MathJax reference.


To learn more, see our tips on writing great answers.




draft saved


draft discarded














StackExchange.ready(
function ()
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f226694%2fimmutable-builder-and-updater%23new-answer', 'question_page');

);

Post as a guest















Required, but never shown





















































Required, but never shown














Required, but never shown












Required, but never shown







Required, but never shown

































Required, but never shown














Required, but never shown












Required, but never shown







Required, but never shown







Popular posts from this blog

Canceling a color specificationRandomly assigning color to Graphics3D objects?Default color for Filling in Mathematica 9Coloring specific elements of sets with a prime modified order in an array plotHow to pick a color differing significantly from the colors already in a given color list?Detection of the text colorColor numbers based on their valueCan color schemes for use with ColorData include opacity specification?My dynamic color schemes

Invision Community Contents History See also References External links Navigation menuProprietaryinvisioncommunity.comIPS Community ForumsIPS Community Forumsthis blog entry"License Changes, IP.Board 3.4, and the Future""Interview -- Matt Mecham of Ibforums""CEO Invision Power Board, Matt Mecham Is a Liar, Thief!"IPB License Explanation 1.3, 1.3.1, 2.0, and 2.1ArchivedSecurity Fixes, Updates And Enhancements For IPB 1.3.1Archived"New Demo Accounts - Invision Power Services"the original"New Default Skin"the original"Invision Power Board 3.0.0 and Applications Released"the original"Archived copy"the original"Perpetual licenses being done away with""Release Notes - Invision Power Services""Introducing: IPS Community Suite 4!"Invision Community Release Notes

199年 目錄 大件事 到箇年出世嗰人 到箇年死嗰人 節慶、風俗習慣 導覽選單