Compare commits
5 Commits
nilmtools-
...
nilmtools-
Author | SHA1 | Date | |
---|---|---|---|
de2a794e00 | |||
065a40f265 | |||
65fa43aff1 | |||
57c23c3792 | |||
d4c8e4acb4 |
@@ -4,15 +4,19 @@ import nilmtools.filter
|
||||
import nilmtools.decimate
|
||||
import nilmdb.client
|
||||
import argparse
|
||||
import fnmatch
|
||||
|
||||
def main(argv = None):
|
||||
parser = argparse.ArgumentParser(
|
||||
formatter_class = argparse.RawDescriptionHelpFormatter,
|
||||
version = "1.0",
|
||||
version = nilmtools.__version__,
|
||||
description = """\
|
||||
Automatically create multiple decimations from a single source
|
||||
stream, continuing until the last decimated level contains fewer
|
||||
than 500 points total.
|
||||
|
||||
Wildcards and multiple paths are accepted. Decimated paths are
|
||||
ignored when matching wildcards.
|
||||
""")
|
||||
parser.add_argument("-u", "--url", action="store",
|
||||
default="http://localhost/nilmdb/",
|
||||
@@ -23,20 +27,36 @@ def main(argv = None):
|
||||
default = False,
|
||||
help="Force metadata changes if the dest "
|
||||
"doesn't match")
|
||||
parser.add_argument("path", action="store",
|
||||
parser.add_argument("path", action="store", nargs='+',
|
||||
help='Path of base stream')
|
||||
args = parser.parse_args(argv)
|
||||
|
||||
# Pull out info about the base stream
|
||||
client = nilmdb.client.Client(args.url)
|
||||
|
||||
info = nilmtools.filter.get_stream_info(client, args.path)
|
||||
if not info:
|
||||
raise Exception("path " + args.path + " not found")
|
||||
# Find list of paths to process
|
||||
streams = [ unicode(s[0]) for s in client.stream_list() ]
|
||||
streams = [ s for s in streams if "~decim-" not in s ]
|
||||
paths = []
|
||||
for path in args.path:
|
||||
new = fnmatch.filter(streams, unicode(path))
|
||||
if not new:
|
||||
print "error: no stream matched path:", path
|
||||
raise SystemExit(1)
|
||||
paths.extend(new)
|
||||
|
||||
meta = client.stream_get_metadata(args.path)
|
||||
for path in paths:
|
||||
do_decimation(client, args, path)
|
||||
|
||||
def do_decimation(client, args, path):
|
||||
print "Decimating", path
|
||||
info = nilmtools.filter.get_stream_info(client, path)
|
||||
if not info:
|
||||
raise Exception("path " + path + " not found")
|
||||
|
||||
meta = client.stream_get_metadata(path)
|
||||
if "decimate_source" in meta:
|
||||
print "Stream", args.path, "was decimated from", meta["decimate_source"]
|
||||
print "Stream", path, "was decimated from", meta["decimate_source"]
|
||||
print "You need to pass the base stream instead"
|
||||
raise SystemExit(1)
|
||||
|
||||
@@ -53,7 +73,7 @@ def main(argv = None):
|
||||
if info.rows <= 500:
|
||||
break
|
||||
factor *= args.factor
|
||||
new_path = "%s~decim-%d" % (args.path, factor)
|
||||
new_path = "%s~decim-%d" % (path, factor)
|
||||
|
||||
# Create the stream if needed
|
||||
new_info = nilmtools.filter.get_stream_info(client, new_path)
|
||||
@@ -72,5 +92,7 @@ def main(argv = None):
|
||||
# Update info using the newly decimated stream
|
||||
info = nilmtools.filter.get_stream_info(client, new_path)
|
||||
|
||||
return
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
|
@@ -77,7 +77,8 @@ def main(argv = None):
|
||||
# Check and set metadata in prep stream
|
||||
f.check_dest_metadata({ "prep_raw_source": f.src.path,
|
||||
"prep_sinefit_source": sinefit.path,
|
||||
"prep_column": args.column })
|
||||
"prep_column": args.column,
|
||||
"prep_rotation": rotation })
|
||||
|
||||
# Run the processing function on all data
|
||||
f.process_numpy(process, args = (client_sinefit, sinefit.path, args.column,
|
||||
|
@@ -18,6 +18,15 @@ def main(argv = None):
|
||||
group.add_argument('-f', '--frequency', action='store', type=float,
|
||||
default=60.0,
|
||||
help='Approximate frequency (default: %(default)s)')
|
||||
group.add_argument('-m', '--min-freq', action='store', type=float,
|
||||
help='Minimum valid frequency '
|
||||
'(default: approximate frequency / 2))')
|
||||
group.add_argument('-M', '--max-freq', action='store', type=float,
|
||||
help='Maximum valid frequency '
|
||||
'(default: approximate frequency * 2))')
|
||||
group.add_argument('-a', '--min-amp', action='store', type=float,
|
||||
default=10.0,
|
||||
help='Minimum signal amplitude (default: %(default)s)')
|
||||
|
||||
# Parse arguments
|
||||
try:
|
||||
@@ -34,13 +43,24 @@ def main(argv = None):
|
||||
parser.error("need a column number >= 1")
|
||||
if args.frequency < 0.1:
|
||||
parser.error("frequency must be >= 0.1")
|
||||
if args.min_freq is None:
|
||||
args.min_freq = args.frequency / 2
|
||||
if args.max_freq is None:
|
||||
args.max_freq = args.frequency * 2
|
||||
if (args.min_freq > args.max_freq or
|
||||
args.min_freq > args.frequency or
|
||||
args.max_freq < args.frequency):
|
||||
parser.error("invalid min or max frequency")
|
||||
if args.min_amp < 0:
|
||||
parser.error("min amplitude must be >= 0")
|
||||
|
||||
f.check_dest_metadata({ "sinefit_source": f.src.path,
|
||||
"sinefit_column": args.column })
|
||||
f.process_numpy(process, args = (args.column, args.frequency))
|
||||
f.process_numpy(process, args = (args.column, args.frequency, args.min_amp,
|
||||
args.min_freq, args.max_freq))
|
||||
|
||||
def process(data, interval, args, insert_function, final):
|
||||
(column, f_expected) = args
|
||||
(column, f_expected, a_min, f_min, f_max) = args
|
||||
rows = data.shape[0]
|
||||
|
||||
# Estimate sampling frequency from timestamps
|
||||
@@ -66,8 +86,14 @@ def process(data, interval, args, insert_function, final):
|
||||
(A, f0, phi, C) = sfit4(this, fs)
|
||||
|
||||
# Check bounds. If frequency is too crazy, ignore this window
|
||||
if f0 < (f_expected/2) or f0 > (f_expected*2):
|
||||
print "frequency", f0, "too far from expected value", f_expected
|
||||
if f0 < f_min or f0 > f_max:
|
||||
print "frequency", f0, "outside valid range", f_min, "-", f_max
|
||||
start += N
|
||||
continue
|
||||
|
||||
# If amplitude is too low, results are probably just noise
|
||||
if A < a_min:
|
||||
print "amplitude", A, "below minimum threshold", a_min
|
||||
start += N
|
||||
continue
|
||||
|
||||
@@ -158,30 +184,30 @@ def sfit4(data, fs):
|
||||
# Convert to Hz
|
||||
f0 = i * fs / N
|
||||
|
||||
## Fit it
|
||||
# first guess for A0, B0 using 3-parameter fit (step c)
|
||||
w = 2*pi*f0
|
||||
D = c_[cos(w*t), sin(w*t), ones(N)]
|
||||
s = linalg.lstsq(D, data)[0]
|
||||
|
||||
# Now iterate 6 times (step i)
|
||||
for idx in range(6):
|
||||
D = c_[cos(w*t), sin(w*t), ones(N),
|
||||
-s[0] * t * sin(w*t) + s[1] * t * cos(w*t) ] # eqn B.16
|
||||
s = linalg.lstsq(D, data)[0] # eqn B.18
|
||||
w = w + s[3] # update frequency estimate
|
||||
|
||||
## Extract results
|
||||
A = sqrt(s[0]*s[0] + s[1]*s[1]) # eqn B.21
|
||||
f0 = w / (2*pi)
|
||||
# Fit it. We'll catch exceptions here and just returns zeros
|
||||
# if something fails with the least squares fit, etc.
|
||||
try:
|
||||
# first guess for A0, B0 using 3-parameter fit (step c)
|
||||
w = 2*pi*f0
|
||||
D = c_[cos(w*t), sin(w*t), ones(N)]
|
||||
s = linalg.lstsq(D, data)[0]
|
||||
|
||||
# Now iterate 6 times (step i)
|
||||
for idx in range(6):
|
||||
D = c_[cos(w*t), sin(w*t), ones(N),
|
||||
-s[0] * t * sin(w*t) + s[1] * t * cos(w*t) ] # eqn B.16
|
||||
s = linalg.lstsq(D, data)[0] # eqn B.18
|
||||
w = w + s[3] # update frequency estimate
|
||||
|
||||
## Extract results
|
||||
A = sqrt(s[0]*s[0] + s[1]*s[1]) # eqn B.21
|
||||
f0 = w / (2*pi)
|
||||
phi = -arctan2(s[1], s[0]) # eqn B.22
|
||||
except TypeError:
|
||||
C = s[2]
|
||||
return (A, f0, phi, C)
|
||||
except Exception as e:
|
||||
# something broke down, just return zeros
|
||||
return (0, 0, 0, 0)
|
||||
C = s[2]
|
||||
|
||||
return (A, f0, phi, C)
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
|
Reference in New Issue
Block a user