Рекурсивно искать свойства в классе, декорированном по атрибуту - PullRequest
0 голосов
/ 14 декабря 2018

Я бы хотел, чтобы все свойства были оформлены атрибутом AggregateAuthorizeIdentifier.Это основной функционал, поэтому он должен быть максимально быстрым.

Я могу искать рекурсивно по всем свойствам, но я бы проверил все системные и сторонние библиотеки.Но это не оптимальное решение.

Сталкивались ли вы с такой проблемой?

public class ReadOrderHistoryQuery
{
    public List<string> Ordering { get; set; }

    public Criteria Criteria { get; set; }

    public class Criteria
    {
        [AggregateAuthorizeIdentifier]
        public int UserId { get; set; }

        public string InvoiceId { get; set; }
    }
}

мое решение:

    static IEnumerable<PropertyWithAttribute> GetAuthorizeIdentifierAttribute(object command)
    {
        var attributedProperties = GetAuthorizeIdentifierAttribute(command.GetType());
        return attributedProperties;
    }
    static IEnumerable<PropertyWithAttribute> GetAuthorizeIdentifierAttribute(Type type)
    {
        //check current property
        var result = type.GetProperties()
            .Where(prop => Attribute.IsDefined(prop, typeof(AggregateAuthorizeIdentifierAttribute)))
            .Select(p => new PropertyWithAttribute()
            {
                Property = p,
                Attribute = p.GetCustomAttribute<AggregateAuthorizeIdentifierAttribute>(true)
            })
            .ToList();
        //deeper check all properties
        result.AddRange(type.GetProperties()
            .SelectMany(p => GetAuthorizeIdentifierAttribute(p.PropertyType)));
        return result;
    }

Ответы [ 2 ]

0 голосов
/ 14 декабря 2018

Finnaly Я получил его с помощью дополнительного селектора типа (по строке).

    const string PropertyTypeStartWithSelector = "Diabdis.";

    static IEnumerable<PropertyWithAttribute> GetAuthorizeIdentifierAttribute(Type type, string propertyPathPrefix)
    {
        var properties = type.GetProperties(BindingFlags.Public | BindingFlags.Instance);
        //check current property
        var result = properties
            .Where(prop => Attribute.IsDefined(prop, typeof(AggregateAuthorizeIdentifierAttribute)))
            .Select(p => new PropertyWithAttribute()
            {
                PropertyPath = propertyPathPrefix == null ? p.Name : $"{propertyPathPrefix}{PropertyPathSeparator}{p.Name}",
                Attribute = p.GetCustomAttribute<AggregateAuthorizeIdentifierAttribute>(true)
            })
            .ToList();

        //check deeper properties, BUT just selector classes
        result.AddRange(properties
            .Where(p => p.DeclaringType.FullName.StartsWith(PropertyTypeStartWithSelector))
            .SelectMany(p => GetAuthorizeIdentifierAttribute(p.PropertyType, propertyPathPrefix == null ? p.Name : $"{propertyPathPrefix}{PropertyPathSeparator}{p.Name}")));
        return result;
    }
0 голосов
/ 14 декабря 2018

Когда мне нужно сделать что-то подобное, я всегда делаю это так:

object objectToSearch = ...

PropertyInfo[] properties = objectToSearch.GetType().GetProperties(BindingFlags.Public | BindingFlags.Instance);
if (properties != null && properties.Length > 0)
{
    properties.ToList().ForEach(p =>
    {
        if (p.GetCustomAttributes(typeof(AggregateAuthorizeIdentifierAttribute), false).Count() == 1)
        {
            // Do something with the property
        }
    });
}

В соответствии с этим вы можете создать метод расширения, который будет возвращать все свойства объекта, который имеет определенныйатрибут

...