src/pyff/mdrepo.py
F

Error: invalid syntax (<unknown>, line 75)</unknown>
Open

            except Exception, ex:
Severity: Minor
Found in src/pyff/mdrepo.py by radon

We encountered an error attempting to analyze this line.

Similar code found in 3 other locations (mass = 44)
Open

        for organizationName in filter_lang(entity.iter("{%s}OrganizationName" % NS['md']), langs=langs):
            info = display
            display = organizationName.text
Severity: Minor
Found in src/pyff/mdrepo.py and 3 other locations by duplication
src/pyff/mdrepo.py on lines 143..145
src/pyff/mdrepo.py on lines 147..149
src/pyff/mdrepo.py on lines 151..153

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 44.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 3 other locations (mass = 44)
Open

        for organizationDisplayName in filter_lang(entity.iter("{%s}OrganizationDisplayName" % NS['md']), langs=langs):
            info = display
            display = organizationDisplayName.text
Severity: Minor
Found in src/pyff/mdrepo.py and 3 other locations by duplication
src/pyff/mdrepo.py on lines 139..141
src/pyff/mdrepo.py on lines 147..149
src/pyff/mdrepo.py on lines 151..153

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 44.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 3 other locations (mass = 44)
Open

        for serviceName in filter_lang(entity.iter("{%s}ServiceName" % NS['md']), langs=langs):
            info = display
            display = serviceName.text
Severity: Minor
Found in src/pyff/mdrepo.py and 3 other locations by duplication
src/pyff/mdrepo.py on lines 139..141
src/pyff/mdrepo.py on lines 143..145
src/pyff/mdrepo.py on lines 151..153

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 44.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 3 other locations (mass = 44)
Open

        for displayName in filter_lang(entity.iter("{%s}DisplayName" % NS['mdui']), langs=langs):
            info = display
            display = displayName.text
Severity: Minor
Found in src/pyff/mdrepo.py and 3 other locations by duplication
src/pyff/mdrepo.py on lines 139..141
src/pyff/mdrepo.py on lines 143..145
src/pyff/mdrepo.py on lines 147..149

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 44.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 1 other location (mass = 39)
Open

        for organizationUrl in filter_lang(entity.iter("{%s}OrganizationURL" % NS['md']), langs=langs):
            info = organizationUrl.text
Severity: Minor
Found in src/pyff/mdrepo.py and 1 other location by duplication
src/pyff/mdrepo.py on lines 158..159

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 39.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 1 other location (mass = 39)
Open

        for description in filter_lang(entity.iter("{%s}Description" % NS['mdui']), langs=langs):
            info = description.text
Severity: Minor
Found in src/pyff/mdrepo.py and 1 other location by duplication
src/pyff/mdrepo.py on lines 155..156

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 39.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 1 other location (mass = 38)
Open

    def is_idp(self, e):
        return bool(e.find(".//{%s}IDPSSODescriptor" % NS['md']) is not None)
Severity: Minor
Found in src/pyff/mdrepo.py and 1 other location by duplication
src/pyff/mdrepo.py on lines 105..106

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 38.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 1 other location (mass = 38)
Open

    def is_sp(self, e):
        return bool(e.find(".//{%s}SPSSODescriptor" % NS['md']) is not None)
Severity: Minor
Found in src/pyff/mdrepo.py and 1 other location by duplication
src/pyff/mdrepo.py on lines 102..103

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 38.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 4 other locations (mass = 37)
Open

    def psu(self, entity, langs):
        for url in filter_lang(entity.iter("{%s}PrivacyStatementURL" % NS['mdui']), langs=langs):
            return url.text
Severity: Minor
Found in src/pyff/mdrepo.py and 4 other locations by duplication
src/pyff/mdrepo.py on lines 171..172
src/pyff/mdrepo.py on lines 174..175
src/pyff/mdrepo.py on lines 177..178
src/pyff/mdrepo.py on lines 180..181

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 37.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 4 other locations (mass = 37)
Open

        for displayName in filter_lang(entity.iter("{%s}DisplayName" % NS['mdui']), langs=langs):
            return displayName.text
Severity: Minor
Found in src/pyff/mdrepo.py and 4 other locations by duplication
src/pyff/mdrepo.py on lines 112..114
src/pyff/mdrepo.py on lines 174..175
src/pyff/mdrepo.py on lines 177..178
src/pyff/mdrepo.py on lines 180..181

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 37.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 4 other locations (mass = 37)
Open

        for serviceName in filter_lang(entity.iter("{%s}ServiceName" % NS['md']), langs=langs):
            return serviceName.text
Severity: Minor
Found in src/pyff/mdrepo.py and 4 other locations by duplication
src/pyff/mdrepo.py on lines 112..114
src/pyff/mdrepo.py on lines 171..172
src/pyff/mdrepo.py on lines 177..178
src/pyff/mdrepo.py on lines 180..181

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 37.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 4 other locations (mass = 37)
Open

        for organizationDisplayName in filter_lang(entity.iter("{%s}OrganizationDisplayName" % NS['md']), langs=langs):
            return organizationDisplayName.text
Severity: Minor
Found in src/pyff/mdrepo.py and 4 other locations by duplication
src/pyff/mdrepo.py on lines 112..114
src/pyff/mdrepo.py on lines 171..172
src/pyff/mdrepo.py on lines 174..175
src/pyff/mdrepo.py on lines 180..181

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 37.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 4 other locations (mass = 37)
Open

        for organizationName in filter_lang(entity.iter("{%s}OrganizationName" % NS['md']), langs=langs):
            return organizationName.text
Severity: Minor
Found in src/pyff/mdrepo.py and 4 other locations by duplication
src/pyff/mdrepo.py on lines 112..114
src/pyff/mdrepo.py on lines 171..172
src/pyff/mdrepo.py on lines 174..175
src/pyff/mdrepo.py on lines 177..178

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 37.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 2 other locations (mass = 36)
Open

        if relt.tag in ('{%s}EntityDescriptor' % NS['md'], '{%s}EntitiesDescriptor' % NS['md']):
Severity: Minor
Found in src/pyff/mdrepo.py and 2 other locations by duplication
src/pyff/mdrepo.py on lines 622..622
src/pyff/mdrepo.py on lines 636..636

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 36.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 2 other locations (mass = 36)
Open

                if relt.tag in ('{%s}XRD' % NS['xrd'], '{%s}XRDS' % NS['xrd']):
Severity: Minor
Found in src/pyff/mdrepo.py and 2 other locations by duplication
src/pyff/mdrepo.py on lines 506..506
src/pyff/mdrepo.py on lines 636..636

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 36.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 2 other locations (mass = 36)
Open

                elif relt.tag in ('{%s}EntityDescriptor' % NS['md'], '{%s}EntitiesDescriptor' % NS['md']):
Severity: Minor
Found in src/pyff/mdrepo.py and 2 other locations by duplication
src/pyff/mdrepo.py on lines 506..506
src/pyff/mdrepo.py on lines 622..622

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 36.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 1 other location (mass = 33)
Open

            if 'http://pyff.io/category/discoverable' in eattr[ATTRS['entity-category']]:
                d['hidden'] = 'false'
Severity: Minor
Found in src/pyff/mdrepo.py and 1 other location by duplication
src/pyff/mdrepo.py on lines 213..219

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 33.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

Similar code found in 1 other location (mass = 33)
Open

        if 'idp' in eattr[ATTRS['role']]:
            d['type'] = 'idp'
            d['hidden'] = 'true'
            if 'http://pyff.io/category/discoverable' in eattr[ATTRS['entity-category']]:
                d['hidden'] = 'false'
Severity: Minor
Found in src/pyff/mdrepo.py and 1 other location by duplication
src/pyff/mdrepo.py on lines 216..217

Duplicated Code

Duplicated code can lead to software that is hard to understand and difficult to change. The Don't Repeat Yourself (DRY) principle states:

Every piece of knowledge must have a single, unambiguous, authoritative representation within a system.

When you violate DRY, bugs and maintenance problems are sure to follow. Duplicated code has a tendency to both continue to replicate and also to diverge (leaving bugs as two similar implementations differ in subtle ways).

Tuning

This issue has a mass of 33.

We set useful threshold defaults for the languages we support but you may want to adjust these settings based on your project guidelines.

The threshold configuration represents the minimum mass a code block must have to be analyzed for duplication. The lower the threshold, the more fine-grained the comparison.

If the engine is too easily reporting duplication, try raising the threshold. If you suspect that the engine isn't catching enough duplication, try lowering the threshold. The best setting tends to differ from language to language.

See codeclimate-duplication's documentation for more information about tuning the mass threshold in your .codeclimate.yml.

Refactorings

Further Reading

There are no issues that match your filters.

Category
Status
"""

This is the implementation of the active repository of SAML metadata. The 'local' and 'remote' pipes operate on this.

"""
import traceback

from pyff.stats import set_metadata_info, get_metadata_info
from pyff.store import entity_attribute_dict

try:
    from cStringIO import StringIO
except ImportError:  # pragma: no cover
    print(" *** install cStringIO for better performance")
    from StringIO import StringIO

from copy import deepcopy
from datetime import datetime
from UserDict import UserDict
import os
import operator

from concurrent import futures
from lxml import etree
from lxml.builder import ElementMaker
from lxml.etree import DocumentInvalid
import xmlsec
import ipaddr
from .constants import ATTRS
from . import merge_strategies
from .logs import log
from .utils import schema, check_signature, filter_lang, root, duration2timedelta, \
    hash_id, MetadataException, find_merge_strategy, entities_list, url2host, subdomains, avg_domain_distance, \
    iter_entities, validate_document, load_url, iso2datetime, xml_error, find_entity
from .constants import NS, NF_URI, EVENT_DROP_ENTITY, EVENT_IMPORT_FAIL


etree.set_default_parser(etree.XMLParser(resolve_entities=False))


class Event(UserDict):
    pass


class Observable(object):
    def __init__(self):
        self.callbacks = []

    def subscribe(self, callback):
        self.callbacks.append(callback)

    def fire(self, **attrs):
        e = Event(attrs)
        e['time'] = datetime.now()
        for fn in self.callbacks:
            fn(e)


def _trunc(x, l):
    return (x[:l] + '..') if len(x) > l else x


class MDRepository(Observable):
    """A class representing a set of SAML Metadata. Instances present as dict-like objects where
    the keys are URIs and values are EntitiesDescriptor elements containing sets of metadata.
    """

    def __init__(self, metadata_cache_enabled=False, min_cache_ttl="PT5M", store=None):
        self.metadata_cache_enabled = metadata_cache_enabled
        self.min_cache_ttl = min_cache_ttl

        if not isinstance(self.min_cache_ttl, int):
            try:
                self.min_cache_ttl = duration2timedelta(self.min_cache_ttl).total_seconds()
            except Exception, ex:
                log.error(ex)
                self.min_cache_ttl = 300
        self.respect_cache_duration = True
        self.default_cache_duration = "PT10M"
        self.retry_limit = 5
        self.store = None

        if store is not None:
            if hasattr(store, '__call__'):
                self.store = store()
            else:
                self.store = store
        else:
            from .store import MemoryStore

            self.store = MemoryStore()
        super(MDRepository, self).__init__()

    def clone(self):
        return MDRepository(metadata_cache_enabled=self.metadata_cache_enabled,
                            min_cache_ttl=self.min_cache_ttl,
                            store=self.store.clone())

    def sha1_id(self, e):
        return hash_id(e, 'sha1')

    def is_idp(self, e):
        return bool(e.find(".//{%s}IDPSSODescriptor" % NS['md']) is not None)

    def is_sp(self, e):
        return bool(e.find(".//{%s}SPSSODescriptor" % NS['md']) is not None)

    def icon(self, e, langs=None):
        for icon in filter_lang(e.iter("{%s}Logo" % NS['mdui']), langs=langs):
            return dict(url=icon.text,width=icon.get('width'),height=icon.get('height'))

    def psu(self, entity, langs):
        for url in filter_lang(entity.iter("{%s}PrivacyStatementURL" % NS['mdui']), langs=langs):
            return url.text

    def geoloc(self, entity):
        for loc in entity.iter("{%s}GeolocationHint" % NS['mdui']):
            pos = loc.text[5:].split(",")
            return dict(lat=pos[0],long=pos[1])

    def domains(self, entity):
        domains = []
        for d in entity.iter("{%s}DomainHint" % NS['mdui']):
            if d.text == '.':
                return []
            domains.append(d.text)
        if not domains:
            domains.append(url2host(entity.get('entityID')))
        return domains

    def ext_display(self, entity, langs=None):
        """Utility-method for computing a displayable string for a given entity.

        :param entity: An EntityDescriptor element
        """
        display = entity.get('entityID')
        info = ''

        for organizationName in filter_lang(entity.iter("{%s}OrganizationName" % NS['md']), langs=langs):
            info = display
            display = organizationName.text

        for organizationDisplayName in filter_lang(entity.iter("{%s}OrganizationDisplayName" % NS['md']), langs=langs):
            info = display
            display = organizationDisplayName.text

        for serviceName in filter_lang(entity.iter("{%s}ServiceName" % NS['md']), langs=langs):
            info = display
            display = serviceName.text

        for displayName in filter_lang(entity.iter("{%s}DisplayName" % NS['mdui']), langs=langs):
            info = display
            display = displayName.text

        for organizationUrl in filter_lang(entity.iter("{%s}OrganizationURL" % NS['md']), langs=langs):
            info = organizationUrl.text

        for description in filter_lang(entity.iter("{%s}Description" % NS['mdui']), langs=langs):
            info = description.text

        if info == entity.get('entityID'):
            info = ''

        return _trunc(display.strip(), 40), _trunc(info.strip(), 256)

    def display(self, entity, langs=None):
        """Utility-method for computing a displayable string for a given entity.

        :param entity: An EntityDescriptor element
        """
        for displayName in filter_lang(entity.iter("{%s}DisplayName" % NS['mdui']), langs=langs):
            return displayName.text

        for serviceName in filter_lang(entity.iter("{%s}ServiceName" % NS['md']), langs=langs):
            return serviceName.text

        for organizationDisplayName in filter_lang(entity.iter("{%s}OrganizationDisplayName" % NS['md']), langs=langs):
            return organizationDisplayName.text

        for organizationName in filter_lang(entity.iter("{%s}OrganizationName" % NS['md']), langs=langs):
            return organizationName.text

        return entity.get('entityID')

    def sub_domains(self, e):
        lst = []
        domains = self.domains(e)
        for d in domains:
            for sub in subdomains(d):
                if not sub in lst:
                    lst.append(sub)
        return lst

    def scopes(self, e):
        elt = e.findall(".//{%s}IDPSSODescriptor/{%s}Extensions/{%s}Scope" % (NS['md'], NS['md'], NS['shibmd']))
        if elt is None or len(elt) == 0:
            return None
        return [s.text for s in elt]

    def discojson(self, e, langs=None):
        if e is None:
            return dict()

        title, descr = self.ext_display(e)
        entity_id = e.get('entityID')

        d = dict(title=title,
                 descr=descr,
                 auth='saml',
                 entityID=entity_id)

        eattr = entity_attribute_dict(e)
        if 'idp' in eattr[ATTRS['role']]:
            d['type'] = 'idp'
            d['hidden'] = 'true'
            if 'http://pyff.io/category/discoverable' in eattr[ATTRS['entity-category']]:
                d['hidden'] = 'false'
        elif 'sp' in eattr[ATTRS['role']]:
            d['type'] = 'sp'

        icon_info = self.icon(e)
        if icon_info is not None:
            d['icon'] = icon_info.get('url', '')
            d['icon_height'] = icon_info.get('height', 64)
            d['icon_width'] = icon_info.get('width', 64)

        scopes = self.scopes(e)
        if scopes is not None and len(scopes) > 0:
            d['scope'] = ",".join(scopes)

        keywords = filter_lang(e.iter("{%s}Keywords" % NS['mdui']), langs=langs)
        if keywords is not None:
            lst = [elt.text for elt in keywords]
            if len(lst) > 0:
                d['keywords'] = ",".join(lst)
        psu = self.psu(e, langs)
        if psu:
            d['psu'] = psu
        geo = self.geoloc(e)
        if geo:
            d['geo'] = geo

        return d

    def simple_summary(self, e):
        if e is None:
            return dict()

        title, descr = self.ext_display(e)
        entity_id = e.get('entityID')
        d = dict(title=title,
                 descr=descr,
                 entityID=entity_id,
                 domains=";".join(self.sub_domains(e)),
                 id=hash_id(e, 'sha1'))
        icon_info = self.icon(e)
        if icon_info is not None:
            url = icon_info.get('url', '')
            d['icon_url'] = url
            d['icon'] = url

        psu = self.psu(e, None)
        if psu:
            d['psu'] = psu

        return d

    def search(self, query=None, path=None, page=None, page_limit=10, entity_filter=None, related=None):
        """
:param query: A string to search for.
:param path: The repository collection (@Name) to search in - None for search in all collections
:param page:  When using paged search, the page index
:param page_limit: When using paged search, the maximum entry per page
:param entity_filter: An optional lookup expression used to filter the entries before search is done.
:param related: an optional '+'-separated list of related domain names for prioritizing search results

Returns a list of dict's for each EntityDescriptor present in the metadata store such
that any of the DisplayName, ServiceName, OrganizationName or OrganizationDisplayName
elements match the query (as in contains the query as a substring).

The dict in the list contains three items:

:param title: A displayable string, useful as a UI label
:param value: The entityID of the EntityDescriptor
:param id: A sha1-ID of the entityID - on the form {sha1}<sha1-hash-of-entityID>
        """

        match_query = bool(len(query) > 0)

        if isinstance(query, basestring):
            query = [query.lower()]

        def _strings(elt):
            lst = []
            for attr in ['{%s}DisplayName' % NS['mdui'],
                         '{%s}ServiceName' % NS['md'],
                         '{%s}OrganizationDisplayName' % NS['md'],
                         '{%s}OrganizationName' % NS['md'],
                         '{%s}Keywords' % NS['mdui'],
                         '{%s}Scope' % NS['shibmd']]:
                lst.extend([s.text for s in elt.iter(attr)])
            lst.append(elt.get('entityID'))
            return filter(lambda item: item is not None, lst)

        def _ip_networks(elt):
            return [ipaddr.IPNetwork(x.text) for x in elt.iter('{%s}IPHint' % NS['mdui'])]

        def _match(qq, elt):
            for q in qq:
                q = q.strip()
                if ':' in q or '.' in q:
                    try:
                        nets = _ip_networks(elt)
                        for net in nets:
                            if ':' in q and ipaddr.IPv6Address(q) in net:
                                return net
                            if '.' in q and ipaddr.IPv4Address(q) in net:
                                return net
                    except ValueError:
                        pass

                if q is not None and len(q) > 0:
                    tokens = _strings(elt)
                    for tstr in tokens:
                        for tpart in tstr.split():
                            if tpart.lower().startswith(q):
                                return tstr
            return None

        f = []
        if path is not None and path not in f:
            f.append(path)
        if entity_filter is not None and entity_filter not in f:
            f.append(entity_filter)
        mexpr = None
        if f:
            mexpr = "+".join(f)

        log.debug("match using '%s'" % mexpr)
        res = []
        for e in self.lookup(mexpr):
            d = None
            #log.debug("query: %s" % query)
            if match_query:
                m = _match(query, e)
                if m is not None:
                    d = self.simple_summary(e)
                    ll = d['title'].lower()
                    if m != ll and not query[0] in ll:
                        d['title'] = "%s - %s" % (d['title'], m)
            else:

                d = self.simple_summary(e)

            if d is not None:
                if related is not None:
                    d['ddist'] = avg_domain_distance(related, d['domains'])
                else:
                    d['ddist'] = 0

                res.append(d)

        res.sort(key=operator.itemgetter('title'))
        res.sort(key=operator.itemgetter('ddist'), reverse=True)

        #log.debug("search returning %s" % res)

        if page is not None:
            total = len(res)
            begin = (page - 1) * page_limit
            end = begin + page_limit
            more = (end < total)
            return res[begin:end], more, total
        else:
            return res

    def sane(self):
        """A very basic test for sanity. An empty metadata set is probably not a sane output of any process.

:return: True iff there is at least one EntityDescriptor in the active set.
        """
        return len(self.store.collections()) > 0

    def extensions(self, e):
        """Return a list of the Extensions elements in the EntityDescriptor

:param e: an EntityDescriptor
:return: a list
        """
        ext = e.find("./{%s}Extensions" % NS['md'])
        if ext is None:
            ext = etree.Element("{%s}Extensions" % NS['md'])
            e.insert(0, ext)
        return ext

    def annotate(self, e, category, title, message, source=None):
        """Add an ATOM annotation to an EntityDescriptor or an EntitiesDescriptor. This is a simple way to
        add non-normative text annotations to metadata, eg for the purpuse of generating reports.

:param e: An EntityDescriptor or an EntitiesDescriptor element
:param category: The ATOM category
:param title: The ATOM title
:param message: The ATOM content
:param source: An optional source URL. It is added as a <link> element with @rel='saml-metadata-source'
        """
        if e.tag != "{%s}EntityDescriptor" % NS['md'] and e.tag != "{%s}EntitiesDescriptor" % NS['md']:
            raise MetadataException("I can only annotate EntityDescriptor or EntitiesDescriptor elements")
        subject = e.get('Name', e.get('entityID', None))
        atom = ElementMaker(nsmap={'atom': 'http://www.w3.org/2005/Atom'}, namespace='http://www.w3.org/2005/Atom')
        args = [atom.published("%s" % datetime.now().isoformat()),
                atom.link(href=subject, rel="saml-metadata-subject")]
        if source is not None:
            args.append(atom.link(href=source, rel="saml-metadata-source"))
        args.extend([atom.title(title),
                     atom.category(term=category),
                     atom.content(message, type="text/plain")])
        self.extensions(e).append(atom.entry(*args))
        self.store.update(e)

    def _entity_attributes(self, e):
        ext = self.extensions(e)
        ea = ext.find(".//{%s}EntityAttributes" % NS['mdattr'])
        if ea is None:
            ea = etree.Element("{%s}EntityAttributes" % NS['mdattr'])
            ext.append(ea)
        return ea

    def _eattribute(self, e, attr, nf):
        ea = self._entity_attributes(e)
        a = ea.xpath(".//saml:Attribute[@NameFormat='%s' and @Name='%s']" % (nf, attr),
                     namespaces=NS,
                     smart_strings=False)
        if a is None or len(a) == 0:
            a = etree.Element("{%s}Attribute" % NS['saml'])
            a.set('NameFormat', nf)
            a.set('Name', attr)
            ea.append(a)
        else:
            a = a[0]
        return a

    def set_entity_attributes(self, e, d, nf=NF_URI):

        """Set an entity attribute on an EntityDescriptor

:param e: The EntityDescriptor element
:param d: A dict of attribute-value pairs that should be added as entity attributes
:param nf: The nameFormat (by default "urn:oasis:names:tc:SAML:2.0:attrname-format:uri") to use.
:raise: MetadataException unless e is an EntityDescriptor element
        """
        if e.tag != "{%s}EntityDescriptor" % NS['md']:
            raise MetadataException("I can only add EntityAttribute(s) to EntityDescriptor elements")

        for attr, value in d.iteritems():
            a = self._eattribute(e, attr, nf)
            velt = etree.Element("{%s}AttributeValue" % NS['saml'])
            velt.text = value
            a.append(velt)

        self.store.update(e)

    def set_pubinfo(self, e, publisher=None, creation_instant=None):
        if e.tag != "{%s}EntitiesDescriptor" % NS['md']:
            raise MetadataException("I can only set RegistrationAuthority to EntitiesDescriptor elements")
        if publisher is None:
            raise MetadataException("At least publisher must be provided")

        if creation_instant is None:
            now = datetime.utcnow()
            creation_instant = now.strftime("%Y-%m-%dT%H:%M:%SZ")

        ext = self.extensions(e)
        pi = ext.find(".//{%s}PublicationInfo" % NS['mdrpi'])
        if pi is not None:
            raise MetadataException("A PublicationInfo element is already present")
        pi = etree.Element("{%s}PublicationInfo" % NS['mdrpi'])
        pi.set('publisher', publisher)
        if creation_instant:
            pi.set('creationInstant', creation_instant)
        ext.append(pi)

    def set_reginfo(self, e, policy=None, authority=None):
        if e.tag != "{%s}EntityDescriptor" % NS['md']:
            raise MetadataException("I can only set RegistrationAuthority to EntityDescriptor elements")
        if authority is None:
            raise MetadataException("At least authority must be provided")
        if policy is None:
            policy = dict()

        ext = self.extensions(e)
        ri = ext.find(".//{%s}RegistrationInfo" % NS['mdrpi'])
        if ri is not None:
            raise MetadataException("A RegistrationInfo element is already present")

        ri = etree.Element("{%s}RegistrationInfo" % NS['mdrpi'])
        ext.append(ri)
        ri.set('registrationAuthority', authority)
        for lang, policy_url in policy.iteritems():
            rp = etree.Element("{%s}RegistrationPolicy" % NS['mdrpi'])
            rp.text = policy_url
            rp.set('{%s}lang' % NS['xml'], lang)
            ri.append(rp)

    def expiration(self, t):
        relt = root(t)
        if relt.tag in ('{%s}EntityDescriptor' % NS['md'], '{%s}EntitiesDescriptor' % NS['md']):
            cache_duration = self.default_cache_duration
            valid_until = relt.get('validUntil', None)
            if valid_until is not None:
                now = datetime.utcnow()
                vu = iso2datetime(valid_until)
                now = now.replace(microsecond=0)
                vu = vu.replace(microsecond=0, tzinfo=None)
                return vu - now
            elif self.respect_cache_duration:
                cache_duration = relt.get('cacheDuration', self.default_cache_duration)
                return duration2timedelta(cache_duration)

        return None

    def fetch_metadata(self, resources, max_workers=5, timeout=120, max_tries=5, validate=False, fail_on_error=False, filter_invalid=True):
        """Fetch a series of metadata URLs and optionally verify signatures.

:param resources: A list of triples (url,cert-or-fingerprint,id, post-callback)
:param max_workers: The maximum number of parallell downloads to run
:param validate: Turn on or off schema validation

The list of triples is processed by first downloading the URL. If a cert-or-fingerprint
is supplied it is used to validate the signature on the received XML. Two forms of XML
is supported: SAML Metadata and XRD.

SAML metadata is (if valid and contains a valid signature) stored under the 'id'
identifier (which defaults to the URL unless provided in the triple.

XRD elements are processed thus: for all <Link> elements that contain a ds;KeyInfo
elements with a X509Certificate and where the <Rel> element contains the string
'urn:oasis:names:tc:SAML:2.0:metadata', the corresponding <URL> element is download
and verified.
        """
        resources = [(url, verifier, tid, post, True) for url, verifier, tid, post in resources]
        return self._fetch_metadata(resources,
                                    max_workers=max_workers,
                                    timeout=timeout,
                                    max_tries=max_tries,
                                    validate=validate,
                                    fail_on_error=fail_on_error,
                                    filter_invalid=filter_invalid )

    def _fetch_metadata(self, resources, max_workers=5, timeout=120, max_tries=5, validate=False, fail_on_error=False, filter_invalid=True):
        tries = dict()

        def _process_url(rurl, verifier, tid, post, enable_cache=True):
            tries.setdefault(rurl, 0)

            try:
                resource = load_url(rurl, timeout=timeout, enable_cache=enable_cache)
            except Exception, ex:
                raise MetadataException(ex, "Exception fetching '%s': %s" % (rurl, str(ex)) )
            if not resource.result:
                raise MetadataException("error fetching '%s'" % rurl)
            xml = resource.result.strip()
            retry_resources = []
            info = {
                'Time Spent': "%s seconds" % resource.time
            }

            tries[rurl] += 1
            info['Tries'] = str(tries[rurl])

            if resource.result is not None:
                info['Bytes'] = str(len(resource.result))
            else:
                raise MetadataException("empty response fetching '%s'" % resource.url)

            info['URL'] = str(rurl)
            info['Cached'] = str(resource.cached)
            info['Date'] = str(resource.date)
            info['Last-Modified'] = str(resource.last_modified)
            info['Validation Errors'] = dict()
            info['Description'] = "Remote metadata"
            info['Status'] = 'success'
            if not validate:
                info['Status'] = 'warning'
                info['Description'] += " (un-validated)"
            if not enable_cache:
                info['Status'] = 'info'

            if resource.resp is not None:
                info['HTTP Response'] = resource.resp

            t, offset = self.parse_metadata(StringIO(xml),
                                            key=verifier,
                                            base_url=rurl,
                                            fail_on_error=fail_on_error,
                                            filter_invalid=filter_invalid,
                                            validate=validate,
                                            validation_errors=info['Validation Errors'],
                                            expiration=self.expiration,
                                            post=post)

            if t is None:
                self.fire(type=EVENT_IMPORT_FAIL, url=rurl)
                raise MetadataException("no valid metadata found at '%s'" % rurl)

            relt = root(t)

            expired = False
            if offset is not None:
                expire_time = datetime.now() + offset
                ttl = offset.total_seconds()
                info['Expiration Time'] = str(expire_time)
                info['Cache TTL'] = str(ttl)
                if ttl < self.min_cache_ttl:
                    if tries[rurl] < max_tries:  # try to get fresh md but we'll use what we have anyway
                        retry_resources.append((rurl, verifier, tid, post, False))
                    else:
                        log.error("giving up on %s" % rurl)
                if ttl < 0:
                    expired = True

            if not expired:
                if relt.tag in ('{%s}XRD' % NS['xrd'], '{%s}XRDS' % NS['xrd']):
                    log.debug("%s looks like an xrd document" % rurl)
                    for xrd in t.iter("{%s}XRD" % NS['xrd']):
                        for link in xrd.findall(".//{%s}Link[@rel='%s']" % (NS['xrd'], NS['md'])):
                            link_href = link.get("href")
                            certs = xmlsec.crypto.CertDict(link)
                            fingerprints = certs.keys()
                            fp = None
                            if len(fingerprints) > 0:
                                fp = fingerprints[0]
                            log.debug("XRD: '%s' verified by '%s'" % (link_href, link))
                            tries.setdefault(link_href, 0)
                            if tries[link_href] < max_tries:
                                retry_resources.append((link_href, fp, link_href, post, True))
                elif relt.tag in ('{%s}EntityDescriptor' % NS['md'], '{%s}EntitiesDescriptor' % NS['md']):
                    n = self.store.update(t, tid)
                    info['Size'] = str(n)
                else:
                    raise MetadataException("unknown metadata type for '%s' (%s)" % (rurl, relt.tag))

            set_metadata_info(tid, info)
            log.debug(info)

            return retry_resources

        while resources:
            with futures.ThreadPoolExecutor(max_workers=max_workers) as executor:
                future_to_url = dict((executor.submit(_process_url, url, verifier, tid, post, enable_cache), url)
                                     for url, verifier, tid, post, enable_cache in resources)

                next_resources = []
                for future in futures.as_completed(future_to_url):
                    url = future_to_url[future]
                    if future.exception() is not None:
                        if fail_on_error:
                            log.error('fetching %r generated an exception' % url)
                            raise future.exception()
                        else:
                            log.error('fetching %s generated an exception: %s' % (url, future.exception()))
                    else:
                        next_resources.extend(future.result())
                resources = next_resources
                log.debug("retrying %s" % resources)

    def filter_invalids(self, t, base_url, validation_errors):
        xsd = schema()
        for e in iter_entities(t):
            if not xsd.validate(e):
                error = xml_error(xsd.error_log, m=base_url)
                entity_id = e.get("entityID")
                log.warn("removing '%s': schema validation failed (%s)" % (entity_id, error))
                validation_errors[entity_id] = error
                if e.getparent() is None:
                    return None
                e.getparent().remove(e)
                self.fire(type=EVENT_DROP_ENTITY, url=base_url, entityID=entity_id, error=error)
        return t

    def parse_metadata(self,
                       source,
                       key=None,
                       base_url=None,
                       fail_on_error=False,
                       filter_invalid=True,
                       validate=True,
                       validation_errors=None,
                       expiration=None,
                       post=None):
        """Parse a piece of XML and return an EntitiesDescriptor element after validation.

:param source: a file-like object containing SAML metadata
:param key: a certificate (file) or a SHA1 fingerprint to use for signature verification
:param base_url: use this base url to resolve relative URLs for XInclude processing
:param fail_on_error: (default: False)
:param filter_invalid: (default True) remove invalid EntityDescriptor elements rather than raise an errror
:param validate: (default: True) set to False to turn off all XML schema validation
:param post: A callable that will be called to modify the parse-tree after schema validation
:param validation_errors: A dict that will be used to return validation errors to the caller
:param expiration: A callable that returns the valid_until datetime of the prsed metadata
(but after xinclude processing and signature validation)
        """

        if validation_errors is None:
            validation_errors = dict()

        try:
            parser = etree.XMLParser(resolve_entities=False)
            t = etree.parse(source, base_url=base_url, parser=parser)
            t.xinclude()

            valid_until = None
            if expiration is not None:
                valid_until = expiration(t)

            t = check_signature(t, key)

            # get rid of ID as early as possible - probably not unique
            for e in iter_entities(t):
                if e.get('ID') is not None:
                    del e.attrib['ID']

            t = root(t)

            if validate:
                if filter_invalid:
                    t = self.filter_invalids(t, base_url=base_url, validation_errors=validation_errors)
                else:  # all or nothing
                    try:
                        validate_document(t)
                    except DocumentInvalid, ex:
                        raise MetadataException("schema validation failed: [%s] '%s': %s" %
                                                (base_url, source, xml_error(ex.error_log, m=base_url)))

            if t is not None:
                if t.tag == "{%s}EntityDescriptor" % NS['md']:
                    t = self.entity_set([t], base_url, copy=False, nsmap=t.nsmap)

                if post is not None:
                    t = post(t)

        except Exception, ex:
            if fail_on_error:
                raise ex
            traceback.print_exc(ex)
            log.error(ex)
            return None, None

        log.debug("returning %d valid entities" % len(list(iter_entities(t))))

        return t, valid_until

    def load_dir(self, directory, ext=".xml", url=None, validate=False, post=None, description=None, fail_on_error=True, filter_invalid=True):
        """
:param directory: A directory to walk.
:param ext: Include files with this extension (default .xml)

Traverse a directory tree looking for metadata. Files ending in the specified extension are included. Directories
starting with '.' are excluded.
        """
        if url is None:
            url = directory

        if description is None:
            description = "All entities found in %s" % directory

        entities = []
        for top, dirs, files in os.walk(directory):
            for dn in dirs:
                if dn.startswith("."):
                    dirs.remove(dn)
            for nm in files:
                if nm.endswith(ext):
                    log.debug("parsing from file %s" % nm)
                    fn = os.path.join(top, nm)
                    try:
                        validation_errors = dict()
                        t, valid_until = self.parse_metadata(fn,
                                                             base_url=url,
                                                             fail_on_error=fail_on_error,
                                                             filter_invalid=filter_invalid,
                                                             validate=validate,
                                                             validation_errors=validation_errors,
                                                             post=post)
                        entities.extend(entities_list(t))  # local metadata is assumed to be ok
                        for (eid, error) in validation_errors.iteritems():
                            log.error(error)
                    except Exception, ex:
                        if fail_on_error:
                            raise MetadataException('Error parsing "%s": %s' % (fn, str(ex)))
                        log.error(ex)

        if entities:
            info = dict(Description=description)
            n = self.store.update(self.entity_set(entities, url, validate=validate, copy=False), url)
            info['Size'] = str(n)
            set_metadata_info(url, info)
        else:
            log.info("no entities found in %s" % directory)

    def find(self, t, member):
        relt = root(t)
        if type(member) is str or type(member) is unicode:
            if '!' in member:
                (src, xp) = member.split("!")
                return relt.xpath(xp, namespaces=NS, smart_strings=False)
            else:
                lst = []
                for e in iter_entities(relt):
                    if e.get('entityID') == member:
                        lst.append(e)
                return lst
        raise MetadataException("unknown format for filtr member: %s" % member)

    def _lookup(self, member):
        if member is None:
            member = "entities"

        if type(member) is str or type(member) is unicode:
            if '!' in member:
                (src, xp) = member.split("!")
                if len(src) == 0:
                    src = None
                return self.lookup(src, xp)

        log.debug("calling store lookup %s" % member)
        return self.store.lookup(member)


    def lookup(self, member, xp=None):
        """
Lookup elements in the working metadata repository

:param member: A selector (cf below)
:type member: basestring
:param xp: An optional xpath filter
:type xp: basestring
:return: An interable of EntityDescriptor elements
:rtype: etree.Element

**Selector Syntax**

    - selector "+" selector
    - [sourceID] "!" xpath
    - attribute=value or {attribute}value
    - entityID
    - source (typically @Name from an EntitiesDescriptor set but could also be an alias)

The first form results in the intersection of the results of doing a lookup on the selectors. The second form
results in the EntityDescriptor elements from the source (defaults to all EntityDescriptors) that match the
xpath expression. The attribute-value forms resuls in the EntityDescriptors that contain the specified entity
attribute pair. If non of these forms apply, the lookup is done using either source ID (normally @Name from
the EntitiesDescriptor) or the entityID of single EntityDescriptors. If member is a URI but isn't part of
the metadata repository then it is fetched an treated as a list of (one per line) of selectors. If all else
fails an empty list is returned.

        """

        l = self._lookup(member)
        if hasattr(l, 'tag'):
            l = [l]
        elif hasattr(l, '__iter__'):
            l = list(l)

        if xp is None:
            return l
        else:
            log.debug("filtering %d entities using xpath %s" % (len(l), xp))
            t = self.entity_set(l, 'dummy')
            if t is None:
                return []
            l = root(t).xpath(xp, namespaces=NS, smart_strings=False)
            log.debug("got %d entities after filtering" % len(l))
            return l

    def entity_set(self, entities, name, lookup_fn=None, cacheDuration=None, validUntil=None, validate=True, copy=True, nsmap=dict()):
        """
:param entities: a set of entities specifiers (lookup is used to find entities from this set)
:param name: the @Name attribute
:param cacheDuration: an XML timedelta expression, eg PT1H for 1hr
:param validUntil: a relative time eg 2w 4d 1h for 2 weeks, 4 days and 1hour from now.

Produce an EntityDescriptors set from a list of entities. Optional Name, cacheDuration and validUntil are affixed.
        """

        #log.debug("entities: %s" % entities)

        if lookup_fn is None:
            lookup_fn = self.lookup

        def _resolve(member, l_fn):
            if hasattr(member, 'tag'):
                return [member]
            else:
                return l_fn(member)

        nsmap.update(NS)
        resolved_entities = set()
        for member in entities:
            for entity in _resolve(member, lookup_fn):
                resolved_entities.add(entity)

        if not resolved_entities:
            return None

        for entity in resolved_entities:
            nsmap.update(entity.nsmap)

        log.debug("selecting %d entities before validation" % len(resolved_entities))

        attrs = dict(Name=name, nsmap=nsmap)
        if cacheDuration is not None:
            attrs['cacheDuration'] = cacheDuration
        if validUntil is not None:
            attrs['validUntil'] = validUntil
        t = etree.Element("{%s}EntitiesDescriptor" % NS['md'], **attrs)
        for entity in resolved_entities:
            entity_id = entity.get('entityID', None)
            if (entity is not None) and (entity_id is not None):
                ent_insert = entity
                if copy:
                    ent_insert = deepcopy(ent_insert)
                t.append(ent_insert)

        if validate:
            try:
                validate_document(t)
            except DocumentInvalid, ex:
                log.debug(xml_error(ex.error_log))
                raise MetadataException("XML schema validation failed: %s" % name)
        return t


    def summary(self, uri):
        """
:param uri: An EntitiesDescriptor URI present in the MDRepository
:return: an information dict

Returns a dict object with basic information about the EntitiesDescriptor
        """
        seen = set()
        info = dict()

        info['Duplicates'] = []
        sz = 0

        for e in self.store.lookup(uri):
            entity_id = e.get('entityID')
            if entity_id in seen:
                info['Duplicates'].append(entity_id)
            else:
                seen.add(entity_id)
                sz += 1

        info['Size'] = str(sz)

        info.update(get_metadata_info(uri))
        if 'Validation Errors' in info and info['Validation Errors']:
            info['Status'] = 'danger'

        info.setdefault('Status', 'default')
        return info

    def delete(self, t):
        """
        :param t: The set of entities to remove from the store

        Completely remove all entities in t from the store.
        """
        for e in iter_entities(t):
            self.store.delete(e)

    def merge(self, t, nt, strategy=merge_strategies.replace_existing, strategy_name=None):
        """
:param t: The EntitiesDescriptor element to merge *into*
:param nt:  The EntitiesDescriptor element to merge *from*
:param strategy: A callable implementing the merge strategy pattern
:param strategy_name: The name of a strategy to import. Overrides the callable if present.
:return:

Two EntitiesDescriptor elements are merged - the second into the first. For each element
in the second collection that is present (using the @entityID attribute as key) in the
first the strategy callable is called with the old and new EntityDescriptor elements
as parameters. The strategy callable thus must implement the following pattern:

:param old_e: The EntityDescriptor from t
:param e: The EntityDescriptor from nt
:return: A merged EntityDescriptor element

Before each call to strategy old_e is removed from the MDRepository index and after
merge the resultant EntityDescriptor is added to the index before it is used to
replace old_e in t.
        """
        if strategy_name is not None:
            strategy = find_merge_strategy(strategy_name)

        for e in iter_entities(nt):
            entity_id = e.get("entityID")
            # we assume ddup:ed tree
            old_e = find_entity(t, entity_id)
            new = strategy(old_e, e)
            if new is not None:
                self.store.update(new)

Size

Lines of code
1003