mirror of
https://github.com/samba-team/samba.git
synced 2024-12-24 21:34:56 +03:00
945 lines
24 KiB
Python
945 lines
24 KiB
Python
|
#! /usr/bin/env python
|
|||
|
# encoding: utf-8
|
|||
|
# WARNING! Do not edit! https://waf.io/book/index.html#_obtaining_the_waf_file
|
|||
|
|
|||
|
#!/usr/bin/env python
|
|||
|
# encoding: utf-8
|
|||
|
# Thomas Nagy, 2005-2016 (ita)
|
|||
|
|
|||
|
"""
|
|||
|
Node: filesystem structure
|
|||
|
|
|||
|
#. Each file/folder is represented by exactly one node.
|
|||
|
|
|||
|
#. Some potential class properties are stored on :py:class:`waflib.Build.BuildContext` : nodes to depend on, etc.
|
|||
|
Unused class members can increase the `.wafpickle` file size sensibly.
|
|||
|
|
|||
|
#. Node objects should never be created directly, use
|
|||
|
the methods :py:func:`Node.make_node` or :py:func:`Node.find_node` for the low-level operations
|
|||
|
|
|||
|
#. The methods :py:func:`Node.find_resource`, :py:func:`Node.find_dir` :py:func:`Node.find_or_declare` must be
|
|||
|
used when a build context is present
|
|||
|
|
|||
|
#. Each instance of :py:class:`waflib.Context.Context` has a unique :py:class:`Node` subclass required for serialization.
|
|||
|
(:py:class:`waflib.Node.Nod3`, see the :py:class:`waflib.Context.Context` initializer). A reference to the context
|
|||
|
owning a node is held as *self.ctx*
|
|||
|
"""
|
|||
|
|
|||
|
import os, re, sys, shutil
|
|||
|
from waflib import Utils, Errors
|
|||
|
|
|||
|
exclude_regs = '''
|
|||
|
**/*~
|
|||
|
**/#*#
|
|||
|
**/.#*
|
|||
|
**/%*%
|
|||
|
**/._*
|
|||
|
**/CVS
|
|||
|
**/CVS/**
|
|||
|
**/.cvsignore
|
|||
|
**/SCCS
|
|||
|
**/SCCS/**
|
|||
|
**/vssver.scc
|
|||
|
**/.svn
|
|||
|
**/.svn/**
|
|||
|
**/BitKeeper
|
|||
|
**/.git
|
|||
|
**/.git/**
|
|||
|
**/.gitignore
|
|||
|
**/.bzr
|
|||
|
**/.bzrignore
|
|||
|
**/.bzr/**
|
|||
|
**/.hg
|
|||
|
**/.hg/**
|
|||
|
**/_MTN
|
|||
|
**/_MTN/**
|
|||
|
**/.arch-ids
|
|||
|
**/{arch}
|
|||
|
**/_darcs
|
|||
|
**/_darcs/**
|
|||
|
**/.intlcache
|
|||
|
**/.DS_Store'''
|
|||
|
"""
|
|||
|
Ant patterns for files and folders to exclude while doing the
|
|||
|
recursive traversal in :py:meth:`waflib.Node.Node.ant_glob`
|
|||
|
"""
|
|||
|
|
|||
|
class Node(object):
|
|||
|
"""
|
|||
|
This class is organized in two parts:
|
|||
|
|
|||
|
* The basic methods meant for filesystem access (compute paths, create folders, etc)
|
|||
|
* The methods bound to a :py:class:`waflib.Build.BuildContext` (require ``bld.srcnode`` and ``bld.bldnode``)
|
|||
|
"""
|
|||
|
|
|||
|
dict_class = dict
|
|||
|
"""
|
|||
|
Subclasses can provide a dict class to enable case insensitivity for example.
|
|||
|
"""
|
|||
|
|
|||
|
__slots__ = ('name', 'parent', 'children', 'cache_abspath', 'cache_isdir')
|
|||
|
def __init__(self, name, parent):
|
|||
|
"""
|
|||
|
.. note:: Use :py:func:`Node.make_node` or :py:func:`Node.find_node` instead of calling this constructor
|
|||
|
"""
|
|||
|
self.name = name
|
|||
|
self.parent = parent
|
|||
|
if parent:
|
|||
|
if name in parent.children:
|
|||
|
raise Errors.WafError('node %s exists in the parent files %r already' % (name, parent))
|
|||
|
parent.children[name] = self
|
|||
|
|
|||
|
def __setstate__(self, data):
|
|||
|
"Deserializes node information, used for persistence"
|
|||
|
self.name = data[0]
|
|||
|
self.parent = data[1]
|
|||
|
if data[2] is not None:
|
|||
|
# Issue 1480
|
|||
|
self.children = self.dict_class(data[2])
|
|||
|
|
|||
|
def __getstate__(self):
|
|||
|
"Serializes node information, used for persistence"
|
|||
|
return (self.name, self.parent, getattr(self, 'children', None))
|
|||
|
|
|||
|
def __str__(self):
|
|||
|
"""
|
|||
|
String representation (abspath), for debugging purposes
|
|||
|
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
return self.abspath()
|
|||
|
|
|||
|
def __repr__(self):
|
|||
|
"""
|
|||
|
String representation (abspath), for debugging purposes
|
|||
|
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
return self.abspath()
|
|||
|
|
|||
|
def __copy__(self):
|
|||
|
"""
|
|||
|
Provided to prevent nodes from being copied
|
|||
|
|
|||
|
:raises: :py:class:`waflib.Errors.WafError`
|
|||
|
"""
|
|||
|
raise Errors.WafError('nodes are not supposed to be copied')
|
|||
|
|
|||
|
def read(self, flags='r', encoding='ISO8859-1'):
|
|||
|
"""
|
|||
|
Reads and returns the contents of the file represented by this node, see :py:func:`waflib.Utils.readf`::
|
|||
|
|
|||
|
def build(bld):
|
|||
|
bld.path.find_node('wscript').read()
|
|||
|
|
|||
|
:param flags: Open mode
|
|||
|
:type flags: string
|
|||
|
:param encoding: encoding value for Python3
|
|||
|
:type encoding: string
|
|||
|
:rtype: string or bytes
|
|||
|
:return: File contents
|
|||
|
"""
|
|||
|
return Utils.readf(self.abspath(), flags, encoding)
|
|||
|
|
|||
|
def write(self, data, flags='w', encoding='ISO8859-1'):
|
|||
|
"""
|
|||
|
Writes data to the file represented by this node, see :py:func:`waflib.Utils.writef`::
|
|||
|
|
|||
|
def build(bld):
|
|||
|
bld.path.make_node('foo.txt').write('Hello, world!')
|
|||
|
|
|||
|
:param data: data to write
|
|||
|
:type data: string
|
|||
|
:param flags: Write mode
|
|||
|
:type flags: string
|
|||
|
:param encoding: encoding value for Python3
|
|||
|
:type encoding: string
|
|||
|
"""
|
|||
|
Utils.writef(self.abspath(), data, flags, encoding)
|
|||
|
|
|||
|
def read_json(self, convert=True, encoding='utf-8'):
|
|||
|
"""
|
|||
|
Reads and parses the contents of this node as JSON (Python ≥ 2.6)::
|
|||
|
|
|||
|
def build(bld):
|
|||
|
bld.path.find_node('abc.json').read_json()
|
|||
|
|
|||
|
Note that this by default automatically decodes unicode strings on Python2, unlike what the Python JSON module does.
|
|||
|
|
|||
|
:type convert: boolean
|
|||
|
:param convert: Prevents decoding of unicode strings on Python2
|
|||
|
:type encoding: string
|
|||
|
:param encoding: The encoding of the file to read. This default to UTF8 as per the JSON standard
|
|||
|
:rtype: object
|
|||
|
:return: Parsed file contents
|
|||
|
"""
|
|||
|
import json # Python 2.6 and up
|
|||
|
object_pairs_hook = None
|
|||
|
if convert and sys.hexversion < 0x3000000:
|
|||
|
try:
|
|||
|
_type = unicode
|
|||
|
except NameError:
|
|||
|
_type = str
|
|||
|
|
|||
|
def convert(value):
|
|||
|
if isinstance(value, list):
|
|||
|
return [convert(element) for element in value]
|
|||
|
elif isinstance(value, _type):
|
|||
|
return str(value)
|
|||
|
else:
|
|||
|
return value
|
|||
|
|
|||
|
def object_pairs(pairs):
|
|||
|
return dict((str(pair[0]), convert(pair[1])) for pair in pairs)
|
|||
|
|
|||
|
object_pairs_hook = object_pairs
|
|||
|
|
|||
|
return json.loads(self.read(encoding=encoding), object_pairs_hook=object_pairs_hook)
|
|||
|
|
|||
|
def write_json(self, data, pretty=True):
|
|||
|
"""
|
|||
|
Writes a python object as JSON to disk (Python ≥ 2.6) as UTF-8 data (JSON standard)::
|
|||
|
|
|||
|
def build(bld):
|
|||
|
bld.path.find_node('xyz.json').write_json(199)
|
|||
|
|
|||
|
:type data: object
|
|||
|
:param data: The data to write to disk
|
|||
|
:type pretty: boolean
|
|||
|
:param pretty: Determines if the JSON will be nicely space separated
|
|||
|
"""
|
|||
|
import json # Python 2.6 and up
|
|||
|
indent = 2
|
|||
|
separators = (',', ': ')
|
|||
|
sort_keys = pretty
|
|||
|
newline = os.linesep
|
|||
|
if not pretty:
|
|||
|
indent = None
|
|||
|
separators = (',', ':')
|
|||
|
newline = ''
|
|||
|
output = json.dumps(data, indent=indent, separators=separators, sort_keys=sort_keys) + newline
|
|||
|
self.write(output, encoding='utf-8')
|
|||
|
|
|||
|
def exists(self):
|
|||
|
"""
|
|||
|
Returns whether the Node is present on the filesystem
|
|||
|
|
|||
|
:rtype: bool
|
|||
|
"""
|
|||
|
return os.path.exists(self.abspath())
|
|||
|
|
|||
|
def isdir(self):
|
|||
|
"""
|
|||
|
Returns whether the Node represents a folder
|
|||
|
|
|||
|
:rtype: bool
|
|||
|
"""
|
|||
|
return os.path.isdir(self.abspath())
|
|||
|
|
|||
|
def chmod(self, val):
|
|||
|
"""
|
|||
|
Changes the file/dir permissions::
|
|||
|
|
|||
|
def build(bld):
|
|||
|
bld.path.chmod(493) # 0755
|
|||
|
"""
|
|||
|
os.chmod(self.abspath(), val)
|
|||
|
|
|||
|
def delete(self, evict=True):
|
|||
|
"""
|
|||
|
Removes the file/folder from the filesystem (equivalent to `rm -rf`), and remove this object from the Node tree.
|
|||
|
Do not use this object after calling this method.
|
|||
|
"""
|
|||
|
try:
|
|||
|
try:
|
|||
|
if os.path.isdir(self.abspath()):
|
|||
|
shutil.rmtree(self.abspath())
|
|||
|
else:
|
|||
|
os.remove(self.abspath())
|
|||
|
except OSError:
|
|||
|
if os.path.exists(self.abspath()):
|
|||
|
raise
|
|||
|
finally:
|
|||
|
if evict:
|
|||
|
self.evict()
|
|||
|
|
|||
|
def evict(self):
|
|||
|
"""
|
|||
|
Removes this node from the Node tree
|
|||
|
"""
|
|||
|
del self.parent.children[self.name]
|
|||
|
|
|||
|
def suffix(self):
|
|||
|
"""
|
|||
|
Returns the file rightmost extension, for example `a.b.c.d → .d`
|
|||
|
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
k = max(0, self.name.rfind('.'))
|
|||
|
return self.name[k:]
|
|||
|
|
|||
|
def height(self):
|
|||
|
"""
|
|||
|
Returns the depth in the folder hierarchy from the filesystem root or from all the file drives
|
|||
|
|
|||
|
:returns: filesystem depth
|
|||
|
:rtype: integer
|
|||
|
"""
|
|||
|
d = self
|
|||
|
val = -1
|
|||
|
while d:
|
|||
|
d = d.parent
|
|||
|
val += 1
|
|||
|
return val
|
|||
|
|
|||
|
def listdir(self):
|
|||
|
"""
|
|||
|
Lists the folder contents
|
|||
|
|
|||
|
:returns: list of file/folder names ordered alphabetically
|
|||
|
:rtype: list of string
|
|||
|
"""
|
|||
|
lst = Utils.listdir(self.abspath())
|
|||
|
lst.sort()
|
|||
|
return lst
|
|||
|
|
|||
|
def mkdir(self):
|
|||
|
"""
|
|||
|
Creates a folder represented by this node. Intermediate folders are created as needed.
|
|||
|
|
|||
|
:raises: :py:class:`waflib.Errors.WafError` when the folder is missing
|
|||
|
"""
|
|||
|
if self.isdir():
|
|||
|
return
|
|||
|
|
|||
|
try:
|
|||
|
self.parent.mkdir()
|
|||
|
except OSError:
|
|||
|
pass
|
|||
|
|
|||
|
if self.name:
|
|||
|
try:
|
|||
|
os.makedirs(self.abspath())
|
|||
|
except OSError:
|
|||
|
pass
|
|||
|
|
|||
|
if not self.isdir():
|
|||
|
raise Errors.WafError('Could not create the directory %r' % self)
|
|||
|
|
|||
|
try:
|
|||
|
self.children
|
|||
|
except AttributeError:
|
|||
|
self.children = self.dict_class()
|
|||
|
|
|||
|
def find_node(self, lst):
|
|||
|
"""
|
|||
|
Finds a node on the file system (files or folders), and creates the corresponding Node objects if it exists
|
|||
|
|
|||
|
:param lst: relative path
|
|||
|
:type lst: string or list of string
|
|||
|
:returns: The corresponding Node object or None if no entry was found on the filesystem
|
|||
|
:rtype: :py:class:´waflib.Node.Node´
|
|||
|
"""
|
|||
|
|
|||
|
if isinstance(lst, str):
|
|||
|
lst = [x for x in Utils.split_path(lst) if x and x != '.']
|
|||
|
|
|||
|
cur = self
|
|||
|
for x in lst:
|
|||
|
if x == '..':
|
|||
|
cur = cur.parent or cur
|
|||
|
continue
|
|||
|
|
|||
|
try:
|
|||
|
ch = cur.children
|
|||
|
except AttributeError:
|
|||
|
cur.children = self.dict_class()
|
|||
|
else:
|
|||
|
try:
|
|||
|
cur = ch[x]
|
|||
|
continue
|
|||
|
except KeyError:
|
|||
|
pass
|
|||
|
|
|||
|
# optimistic: create the node first then look if it was correct to do so
|
|||
|
cur = self.__class__(x, cur)
|
|||
|
if not cur.exists():
|
|||
|
cur.evict()
|
|||
|
return None
|
|||
|
|
|||
|
if not cur.exists():
|
|||
|
cur.evict()
|
|||
|
return None
|
|||
|
|
|||
|
return cur
|
|||
|
|
|||
|
def make_node(self, lst):
|
|||
|
"""
|
|||
|
Returns or creates a Node object corresponding to the input path without considering the filesystem.
|
|||
|
|
|||
|
:param lst: relative path
|
|||
|
:type lst: string or list of string
|
|||
|
:rtype: :py:class:´waflib.Node.Node´
|
|||
|
"""
|
|||
|
if isinstance(lst, str):
|
|||
|
lst = [x for x in Utils.split_path(lst) if x and x != '.']
|
|||
|
|
|||
|
cur = self
|
|||
|
for x in lst:
|
|||
|
if x == '..':
|
|||
|
cur = cur.parent or cur
|
|||
|
continue
|
|||
|
|
|||
|
try:
|
|||
|
cur = cur.children[x]
|
|||
|
except AttributeError:
|
|||
|
cur.children = self.dict_class()
|
|||
|
except KeyError:
|
|||
|
pass
|
|||
|
else:
|
|||
|
continue
|
|||
|
cur = self.__class__(x, cur)
|
|||
|
return cur
|
|||
|
|
|||
|
def search_node(self, lst):
|
|||
|
"""
|
|||
|
Returns a Node previously defined in the data structure. The filesystem is not considered.
|
|||
|
|
|||
|
:param lst: relative path
|
|||
|
:type lst: string or list of string
|
|||
|
:rtype: :py:class:´waflib.Node.Node´ or None if there is no entry in the Node datastructure
|
|||
|
"""
|
|||
|
if isinstance(lst, str):
|
|||
|
lst = [x for x in Utils.split_path(lst) if x and x != '.']
|
|||
|
|
|||
|
cur = self
|
|||
|
for x in lst:
|
|||
|
if x == '..':
|
|||
|
cur = cur.parent or cur
|
|||
|
else:
|
|||
|
try:
|
|||
|
cur = cur.children[x]
|
|||
|
except (AttributeError, KeyError):
|
|||
|
return None
|
|||
|
return cur
|
|||
|
|
|||
|
def path_from(self, node):
|
|||
|
"""
|
|||
|
Path of this node seen from the other::
|
|||
|
|
|||
|
def build(bld):
|
|||
|
n1 = bld.path.find_node('foo/bar/xyz.txt')
|
|||
|
n2 = bld.path.find_node('foo/stuff/')
|
|||
|
n1.path_from(n2) # '../bar/xyz.txt'
|
|||
|
|
|||
|
:param node: path to use as a reference
|
|||
|
:type node: :py:class:`waflib.Node.Node`
|
|||
|
:returns: a relative path or an absolute one if that is better
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
c1 = self
|
|||
|
c2 = node
|
|||
|
|
|||
|
c1h = c1.height()
|
|||
|
c2h = c2.height()
|
|||
|
|
|||
|
lst = []
|
|||
|
up = 0
|
|||
|
|
|||
|
while c1h > c2h:
|
|||
|
lst.append(c1.name)
|
|||
|
c1 = c1.parent
|
|||
|
c1h -= 1
|
|||
|
|
|||
|
while c2h > c1h:
|
|||
|
up += 1
|
|||
|
c2 = c2.parent
|
|||
|
c2h -= 1
|
|||
|
|
|||
|
while not c1 is c2:
|
|||
|
lst.append(c1.name)
|
|||
|
up += 1
|
|||
|
|
|||
|
c1 = c1.parent
|
|||
|
c2 = c2.parent
|
|||
|
|
|||
|
if c1.parent:
|
|||
|
lst.extend(['..'] * up)
|
|||
|
lst.reverse()
|
|||
|
return os.sep.join(lst) or '.'
|
|||
|
else:
|
|||
|
return self.abspath()
|
|||
|
|
|||
|
def abspath(self):
|
|||
|
"""
|
|||
|
Returns the absolute path. A cache is kept in the context as ``cache_node_abspath``
|
|||
|
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
try:
|
|||
|
return self.cache_abspath
|
|||
|
except AttributeError:
|
|||
|
pass
|
|||
|
# think twice before touching this (performance + complexity + correctness)
|
|||
|
|
|||
|
if not self.parent:
|
|||
|
val = os.sep
|
|||
|
elif not self.parent.name:
|
|||
|
val = os.sep + self.name
|
|||
|
else:
|
|||
|
val = self.parent.abspath() + os.sep + self.name
|
|||
|
self.cache_abspath = val
|
|||
|
return val
|
|||
|
|
|||
|
if Utils.is_win32:
|
|||
|
def abspath(self):
|
|||
|
try:
|
|||
|
return self.cache_abspath
|
|||
|
except AttributeError:
|
|||
|
pass
|
|||
|
if not self.parent:
|
|||
|
val = ''
|
|||
|
elif not self.parent.name:
|
|||
|
val = self.name + os.sep
|
|||
|
else:
|
|||
|
val = self.parent.abspath().rstrip(os.sep) + os.sep + self.name
|
|||
|
self.cache_abspath = val
|
|||
|
return val
|
|||
|
|
|||
|
def is_child_of(self, node):
|
|||
|
"""
|
|||
|
Returns whether the object belongs to a subtree of the input node::
|
|||
|
|
|||
|
def build(bld):
|
|||
|
node = bld.path.find_node('wscript')
|
|||
|
node.is_child_of(bld.path) # True
|
|||
|
|
|||
|
:param node: path to use as a reference
|
|||
|
:type node: :py:class:`waflib.Node.Node`
|
|||
|
:rtype: bool
|
|||
|
"""
|
|||
|
p = self
|
|||
|
diff = self.height() - node.height()
|
|||
|
while diff > 0:
|
|||
|
diff -= 1
|
|||
|
p = p.parent
|
|||
|
return p is node
|
|||
|
|
|||
|
def ant_iter(self, accept=None, maxdepth=25, pats=[], dir=False, src=True, remove=True):
|
|||
|
"""
|
|||
|
Recursive method used by :py:meth:`waflib.Node.ant_glob`.
|
|||
|
|
|||
|
:param accept: function used for accepting/rejecting a node, returns the patterns that can be still accepted in recursion
|
|||
|
:type accept: function
|
|||
|
:param maxdepth: maximum depth in the filesystem (25)
|
|||
|
:type maxdepth: int
|
|||
|
:param pats: list of patterns to accept and list of patterns to exclude
|
|||
|
:type pats: tuple
|
|||
|
:param dir: return folders too (False by default)
|
|||
|
:type dir: bool
|
|||
|
:param src: return files (True by default)
|
|||
|
:type src: bool
|
|||
|
:param remove: remove files/folders that do not exist (True by default)
|
|||
|
:type remove: bool
|
|||
|
:returns: A generator object to iterate from
|
|||
|
:rtype: iterator
|
|||
|
"""
|
|||
|
dircont = self.listdir()
|
|||
|
dircont.sort()
|
|||
|
|
|||
|
try:
|
|||
|
lst = set(self.children.keys())
|
|||
|
except AttributeError:
|
|||
|
self.children = self.dict_class()
|
|||
|
else:
|
|||
|
if remove:
|
|||
|
for x in lst - set(dircont):
|
|||
|
self.children[x].evict()
|
|||
|
|
|||
|
for name in dircont:
|
|||
|
npats = accept(name, pats)
|
|||
|
if npats and npats[0]:
|
|||
|
accepted = [] in npats[0]
|
|||
|
|
|||
|
node = self.make_node([name])
|
|||
|
|
|||
|
isdir = node.isdir()
|
|||
|
if accepted:
|
|||
|
if isdir:
|
|||
|
if dir:
|
|||
|
yield node
|
|||
|
else:
|
|||
|
if src:
|
|||
|
yield node
|
|||
|
|
|||
|
if isdir:
|
|||
|
node.cache_isdir = True
|
|||
|
if maxdepth:
|
|||
|
for k in node.ant_iter(accept=accept, maxdepth=maxdepth - 1, pats=npats, dir=dir, src=src, remove=remove):
|
|||
|
yield k
|
|||
|
raise StopIteration
|
|||
|
|
|||
|
def ant_glob(self, *k, **kw):
|
|||
|
"""
|
|||
|
Finds files across folders:
|
|||
|
|
|||
|
* ``**/*`` find all files recursively
|
|||
|
* ``**/*.class`` find all files ending by .class
|
|||
|
* ``..`` find files having two dot characters
|
|||
|
|
|||
|
For example::
|
|||
|
|
|||
|
def configure(cfg):
|
|||
|
cfg.path.ant_glob('**/*.cpp') # finds all .cpp files
|
|||
|
cfg.root.ant_glob('etc/*.txt') # matching from the filesystem root can be slow
|
|||
|
cfg.path.ant_glob('*.cpp', excl=['*.c'], src=True, dir=False)
|
|||
|
|
|||
|
For more information see http://ant.apache.org/manual/dirtasks.html
|
|||
|
|
|||
|
The nodes that correspond to files and folders that do not exist are garbage-collected.
|
|||
|
To prevent this behaviour in particular when running over the build directory, pass ``remove=False``
|
|||
|
|
|||
|
:param incl: ant patterns or list of patterns to include
|
|||
|
:type incl: string or list of strings
|
|||
|
:param excl: ant patterns or list of patterns to exclude
|
|||
|
:type excl: string or list of strings
|
|||
|
:param dir: return folders too (False by default)
|
|||
|
:type dir: bool
|
|||
|
:param src: return files (True by default)
|
|||
|
:type src: bool
|
|||
|
:param remove: remove files/folders that do not exist (True by default)
|
|||
|
:type remove: bool
|
|||
|
:param maxdepth: maximum depth of recursion
|
|||
|
:type maxdepth: int
|
|||
|
:param ignorecase: ignore case while matching (False by default)
|
|||
|
:type ignorecase: bool
|
|||
|
:returns: The corresponding Nodes
|
|||
|
:rtype: list of :py:class:`waflib.Node.Node` instances
|
|||
|
"""
|
|||
|
|
|||
|
src = kw.get('src', True)
|
|||
|
dir = kw.get('dir', False)
|
|||
|
|
|||
|
excl = kw.get('excl', exclude_regs)
|
|||
|
incl = k and k[0] or kw.get('incl', '**')
|
|||
|
reflags = kw.get('ignorecase', 0) and re.I
|
|||
|
|
|||
|
def to_pat(s):
|
|||
|
lst = Utils.to_list(s)
|
|||
|
ret = []
|
|||
|
for x in lst:
|
|||
|
x = x.replace('\\', '/').replace('//', '/')
|
|||
|
if x.endswith('/'):
|
|||
|
x += '**'
|
|||
|
lst2 = x.split('/')
|
|||
|
accu = []
|
|||
|
for k in lst2:
|
|||
|
if k == '**':
|
|||
|
accu.append(k)
|
|||
|
else:
|
|||
|
k = k.replace('.', '[.]').replace('*','.*').replace('?', '.').replace('+', '\\+')
|
|||
|
k = '^%s$' % k
|
|||
|
try:
|
|||
|
#print "pattern", k
|
|||
|
accu.append(re.compile(k, flags=reflags))
|
|||
|
except Exception ,e:
|
|||
|
raise Errors.WafError('Invalid pattern: %s' % k, e)
|
|||
|
ret.append(accu)
|
|||
|
return ret
|
|||
|
|
|||
|
def filtre(name, nn):
|
|||
|
ret = []
|
|||
|
for lst in nn:
|
|||
|
if not lst:
|
|||
|
pass
|
|||
|
elif lst[0] == '**':
|
|||
|
ret.append(lst)
|
|||
|
if len(lst) > 1:
|
|||
|
if lst[1].match(name):
|
|||
|
ret.append(lst[2:])
|
|||
|
else:
|
|||
|
ret.append([])
|
|||
|
elif lst[0].match(name):
|
|||
|
ret.append(lst[1:])
|
|||
|
return ret
|
|||
|
|
|||
|
def accept(name, pats):
|
|||
|
nacc = filtre(name, pats[0])
|
|||
|
nrej = filtre(name, pats[1])
|
|||
|
if [] in nrej:
|
|||
|
nacc = []
|
|||
|
return [nacc, nrej]
|
|||
|
|
|||
|
ret = [x for x in self.ant_iter(accept=accept, pats=[to_pat(incl), to_pat(excl)], maxdepth=kw.get('maxdepth', 25), dir=dir, src=src, remove=kw.get('remove', True))]
|
|||
|
if kw.get('flat', False):
|
|||
|
return ' '.join([x.path_from(self) for x in ret])
|
|||
|
|
|||
|
return ret
|
|||
|
|
|||
|
# --------------------------------------------------------------------------------
|
|||
|
# the following methods require the source/build folders (bld.srcnode/bld.bldnode)
|
|||
|
# using a subclass is a possibility, but is that really necessary?
|
|||
|
# --------------------------------------------------------------------------------
|
|||
|
|
|||
|
def is_src(self):
|
|||
|
"""
|
|||
|
Returns True if the node is below the source directory. Note that ``!is_src() ≠ is_bld()``
|
|||
|
|
|||
|
:rtype: bool
|
|||
|
"""
|
|||
|
cur = self
|
|||
|
x = self.ctx.srcnode
|
|||
|
y = self.ctx.bldnode
|
|||
|
while cur.parent:
|
|||
|
if cur is y:
|
|||
|
return False
|
|||
|
if cur is x:
|
|||
|
return True
|
|||
|
cur = cur.parent
|
|||
|
return False
|
|||
|
|
|||
|
def is_bld(self):
|
|||
|
"""
|
|||
|
Returns True if the node is below the build directory. Note that ``!is_bld() ≠ is_src()``
|
|||
|
|
|||
|
:rtype: bool
|
|||
|
"""
|
|||
|
cur = self
|
|||
|
y = self.ctx.bldnode
|
|||
|
while cur.parent:
|
|||
|
if cur is y:
|
|||
|
return True
|
|||
|
cur = cur.parent
|
|||
|
return False
|
|||
|
|
|||
|
def get_src(self):
|
|||
|
"""
|
|||
|
Returns the corresponding Node object in the source directory (or self if already
|
|||
|
under the source directory). Use this method only if the purpose is to create
|
|||
|
a Node object (this is common with folders but not with files, see ticket 1937)
|
|||
|
|
|||
|
:rtype: :py:class:`waflib.Node.Node`
|
|||
|
"""
|
|||
|
cur = self
|
|||
|
x = self.ctx.srcnode
|
|||
|
y = self.ctx.bldnode
|
|||
|
lst = []
|
|||
|
while cur.parent:
|
|||
|
if cur is y:
|
|||
|
lst.reverse()
|
|||
|
return x.make_node(lst)
|
|||
|
if cur is x:
|
|||
|
return self
|
|||
|
lst.append(cur.name)
|
|||
|
cur = cur.parent
|
|||
|
return self
|
|||
|
|
|||
|
def get_bld(self):
|
|||
|
"""
|
|||
|
Return the corresponding Node object in the build directory (or self if already
|
|||
|
under the build directory). Use this method only if the purpose is to create
|
|||
|
a Node object (this is common with folders but not with files, see ticket 1937)
|
|||
|
|
|||
|
:rtype: :py:class:`waflib.Node.Node`
|
|||
|
"""
|
|||
|
cur = self
|
|||
|
x = self.ctx.srcnode
|
|||
|
y = self.ctx.bldnode
|
|||
|
lst = []
|
|||
|
while cur.parent:
|
|||
|
if cur is y:
|
|||
|
return self
|
|||
|
if cur is x:
|
|||
|
lst.reverse()
|
|||
|
return self.ctx.bldnode.make_node(lst)
|
|||
|
lst.append(cur.name)
|
|||
|
cur = cur.parent
|
|||
|
# the file is external to the current project, make a fake root in the current build directory
|
|||
|
lst.reverse()
|
|||
|
if lst and Utils.is_win32 and len(lst[0]) == 2 and lst[0].endswith(':'):
|
|||
|
lst[0] = lst[0][0]
|
|||
|
return self.ctx.bldnode.make_node(['__root__'] + lst)
|
|||
|
|
|||
|
def find_resource(self, lst):
|
|||
|
"""
|
|||
|
Use this method in the build phase to find source files corresponding to the relative path given.
|
|||
|
|
|||
|
First it looks up the Node data structure to find any declared Node object in the build directory.
|
|||
|
If None is found, it then considers the filesystem in the source directory.
|
|||
|
|
|||
|
:param lst: relative path
|
|||
|
:type lst: string or list of string
|
|||
|
:returns: the corresponding Node object or None
|
|||
|
:rtype: :py:class:`waflib.Node.Node`
|
|||
|
"""
|
|||
|
if isinstance(lst, str):
|
|||
|
lst = [x for x in Utils.split_path(lst) if x and x != '.']
|
|||
|
|
|||
|
node = self.get_bld().search_node(lst)
|
|||
|
if not node:
|
|||
|
node = self.get_src().find_node(lst)
|
|||
|
if node and node.isdir():
|
|||
|
return None
|
|||
|
return node
|
|||
|
|
|||
|
def find_or_declare(self, lst):
|
|||
|
"""
|
|||
|
Use this method in the build phase to declare output files.
|
|||
|
|
|||
|
If 'self' is in build directory, it first tries to return an existing node object.
|
|||
|
If no Node is found, it tries to find one in the source directory.
|
|||
|
If no Node is found, a new Node object is created in the build directory, and the
|
|||
|
intermediate folders are added.
|
|||
|
|
|||
|
:param lst: relative path
|
|||
|
:type lst: string or list of string
|
|||
|
"""
|
|||
|
if isinstance(lst, str):
|
|||
|
lst = [x for x in Utils.split_path(lst) if x and x != '.']
|
|||
|
|
|||
|
node = self.get_bld().search_node(lst)
|
|||
|
if node:
|
|||
|
if not os.path.isfile(node.abspath()):
|
|||
|
node.parent.mkdir()
|
|||
|
return node
|
|||
|
self = self.get_src()
|
|||
|
node = self.find_node(lst)
|
|||
|
if node:
|
|||
|
return node
|
|||
|
node = self.get_bld().make_node(lst)
|
|||
|
node.parent.mkdir()
|
|||
|
return node
|
|||
|
|
|||
|
def find_dir(self, lst):
|
|||
|
"""
|
|||
|
Searches for a folder on the filesystem (see :py:meth:`waflib.Node.Node.find_node`)
|
|||
|
|
|||
|
:param lst: relative path
|
|||
|
:type lst: string or list of string
|
|||
|
:returns: The corresponding Node object or None if there is no such folder
|
|||
|
:rtype: :py:class:`waflib.Node.Node`
|
|||
|
"""
|
|||
|
if isinstance(lst, str):
|
|||
|
lst = [x for x in Utils.split_path(lst) if x and x != '.']
|
|||
|
|
|||
|
node = self.find_node(lst)
|
|||
|
if node and not node.isdir():
|
|||
|
return None
|
|||
|
return node
|
|||
|
|
|||
|
# helpers for building things
|
|||
|
def change_ext(self, ext, ext_in=None):
|
|||
|
"""
|
|||
|
Declares a build node with a distinct extension; this is uses :py:meth:`waflib.Node.Node.find_or_declare`
|
|||
|
|
|||
|
:return: A build node of the same path, but with a different extension
|
|||
|
:rtype: :py:class:`waflib.Node.Node`
|
|||
|
"""
|
|||
|
name = self.name
|
|||
|
if ext_in is None:
|
|||
|
k = name.rfind('.')
|
|||
|
if k >= 0:
|
|||
|
name = name[:k] + ext
|
|||
|
else:
|
|||
|
name = name + ext
|
|||
|
else:
|
|||
|
name = name[:- len(ext_in)] + ext
|
|||
|
|
|||
|
return self.parent.find_or_declare([name])
|
|||
|
|
|||
|
def bldpath(self):
|
|||
|
"""
|
|||
|
Returns the relative path seen from the build directory ``src/foo.cpp``
|
|||
|
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
return self.path_from(self.ctx.bldnode)
|
|||
|
|
|||
|
def srcpath(self):
|
|||
|
"""
|
|||
|
Returns the relative path seen from the source directory ``../src/foo.cpp``
|
|||
|
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
return self.path_from(self.ctx.srcnode)
|
|||
|
|
|||
|
def relpath(self):
|
|||
|
"""
|
|||
|
If a file in the build directory, returns :py:meth:`waflib.Node.Node.bldpath`,
|
|||
|
else returns :py:meth:`waflib.Node.Node.srcpath`
|
|||
|
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
cur = self
|
|||
|
x = self.ctx.bldnode
|
|||
|
while cur.parent:
|
|||
|
if cur is x:
|
|||
|
return self.bldpath()
|
|||
|
cur = cur.parent
|
|||
|
return self.srcpath()
|
|||
|
|
|||
|
def bld_dir(self):
|
|||
|
"""
|
|||
|
Equivalent to self.parent.bldpath()
|
|||
|
|
|||
|
:rtype: string
|
|||
|
"""
|
|||
|
return self.parent.bldpath()
|
|||
|
|
|||
|
def h_file(self):
|
|||
|
"""
|
|||
|
See :py:func:`waflib.Utils.h_file`
|
|||
|
|
|||
|
:return: a hash representing the file contents
|
|||
|
:rtype: string or bytes
|
|||
|
"""
|
|||
|
return Utils.h_file(self.abspath())
|
|||
|
|
|||
|
def get_bld_sig(self):
|
|||
|
"""
|
|||
|
Returns a signature (see :py:meth:`waflib.Node.Node.h_file`) for the purpose
|
|||
|
of build dependency calculation. This method uses a per-context cache.
|
|||
|
|
|||
|
:return: a hash representing the object contents
|
|||
|
:rtype: string or bytes
|
|||
|
"""
|
|||
|
# previous behaviour can be set by returning self.ctx.node_sigs[self] when a build node
|
|||
|
try:
|
|||
|
cache = self.ctx.cache_sig
|
|||
|
except AttributeError:
|
|||
|
cache = self.ctx.cache_sig = {}
|
|||
|
try:
|
|||
|
ret = cache[self]
|
|||
|
except KeyError:
|
|||
|
p = self.abspath()
|
|||
|
try:
|
|||
|
ret = cache[self] = self.h_file()
|
|||
|
except EnvironmentError:
|
|||
|
if self.isdir():
|
|||
|
# allow folders as build nodes, do not use the creation time
|
|||
|
st = os.stat(p)
|
|||
|
ret = cache[self] = Utils.h_list([p, st.st_ino, st.st_mode])
|
|||
|
return ret
|
|||
|
raise
|
|||
|
return ret
|
|||
|
|
|||
|
# --------------------------------------------
|
|||
|
# TODO waf 2.0, remove the sig and cache_sig attributes
|
|||
|
def get_sig(self):
|
|||
|
return self.h_file()
|
|||
|
def set_sig(self, val):
|
|||
|
# clear the cache, so that past implementation should still work
|
|||
|
try:
|
|||
|
del self.get_bld_sig.__cache__[(self,)]
|
|||
|
except (AttributeError, KeyError):
|
|||
|
pass
|
|||
|
sig = property(get_sig, set_sig)
|
|||
|
cache_sig = property(get_sig, set_sig)
|
|||
|
|
|||
|
pickle_lock = Utils.threading.Lock()
|
|||
|
"""Lock mandatory for thread-safe node serialization"""
|
|||
|
|
|||
|
class Nod3(Node):
|
|||
|
"""Mandatory subclass for thread-safe node serialization"""
|
|||
|
pass # do not remove
|