forked from shaba/openuds
Merge remote-tracking branch 'origin/v3.5'
This commit is contained in:
commit
1fb8956679
@ -131,7 +131,7 @@ class UDSApi: # pylint: disable=too-few-public-methods
|
|||||||
headers=headers,
|
headers=headers,
|
||||||
verify=self._validateCert,
|
verify=self._validateCert,
|
||||||
timeout=TIMEOUT,
|
timeout=TIMEOUT,
|
||||||
proxies=NO_PROXY
|
proxies=NO_PROXY # type: ignore
|
||||||
if disableProxy
|
if disableProxy
|
||||||
else None, # if not proxies wanted, enforce it
|
else None, # if not proxies wanted, enforce it
|
||||||
)
|
)
|
||||||
|
@ -528,7 +528,7 @@ class gui:
|
|||||||
|
|
||||||
def date(self, min: bool = True) -> datetime.date:
|
def date(self, min: bool = True) -> datetime.date:
|
||||||
"""
|
"""
|
||||||
Returns the date tis objecct represents
|
Returns the date this object represents
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
min (bool, optional): If true, in case of invalid date will return "min" date, else "max". Defaults to True.
|
min (bool, optional): If true, in case of invalid date will return "min" date, else "max". Defaults to True.
|
||||||
@ -543,6 +543,23 @@ class gui:
|
|||||||
except Exception:
|
except Exception:
|
||||||
return datetime.date.min if min else datetime.date.max
|
return datetime.date.min if min else datetime.date.max
|
||||||
|
|
||||||
|
def datetime(self, min: bool) -> datetime.datetime:
|
||||||
|
"""
|
||||||
|
Returns the date this object represents
|
||||||
|
|
||||||
|
Args:
|
||||||
|
min (bool, optional): If true, in case of invalid date will return "min" date, else "max". Defaults to True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
datetime.date: the date that this object holds, or "min" | "max" on error
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
return datetime.datetime.strptime(
|
||||||
|
self.value, '%Y-%m-%d'
|
||||||
|
) # ISO Format
|
||||||
|
except Exception:
|
||||||
|
return datetime.datetime.min if min else datetime.datetime.max
|
||||||
|
|
||||||
def stamp(self) -> int:
|
def stamp(self) -> int:
|
||||||
return int(
|
return int(
|
||||||
time.mktime(
|
time.mktime(
|
||||||
|
@ -118,11 +118,12 @@ class StatsCounters(models.Model):
|
|||||||
# Max intervals, if present, will adjust interval (that are seconds)
|
# Max intervals, if present, will adjust interval (that are seconds)
|
||||||
max_intervals = kwargs.get('max_intervals', 0)
|
max_intervals = kwargs.get('max_intervals', 0)
|
||||||
if max_intervals > 0:
|
if max_intervals > 0:
|
||||||
interval = max(interval, int(to - since) / max_intervals)
|
count = q.count()
|
||||||
|
if max_intervals < count:
|
||||||
|
max_intervals = count
|
||||||
|
interval = int(to - since) / max_intervals
|
||||||
|
|
||||||
floor = getSqlFnc('FLOOR')
|
floor = getSqlFnc('FLOOR')
|
||||||
ceil = getSqlFnc('CEIL')
|
|
||||||
avg = getSqlFnc('AVG')
|
|
||||||
if interval > 0:
|
if interval > 0:
|
||||||
q = q.extra(
|
q = q.extra(
|
||||||
select={
|
select={
|
||||||
|
@ -57,6 +57,7 @@ reportAutoModelDct: typing.Mapping[str, typing.Type[ReportAutoModel]] = { # typ
|
|||||||
'Provider': models.Provider,
|
'Provider': models.Provider,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class ReportAutoType(UserInterfaceType):
|
class ReportAutoType(UserInterfaceType):
|
||||||
def __new__(cls, name, bases, attrs) -> 'ReportAutoType':
|
def __new__(cls, name, bases, attrs) -> 'ReportAutoType':
|
||||||
# Add gui for elements...
|
# Add gui for elements...
|
||||||
@ -141,3 +142,56 @@ class ReportAuto(Report, metaclass=ReportAutoType):
|
|||||||
return {'hour': 1, 'day': 24, 'week': 24 * 7, 'month': 24 * 30}[
|
return {'hour': 1, 'day': 24, 'week': 24 * 7, 'month': 24 * 30}[
|
||||||
self.interval.value
|
self.interval.value
|
||||||
]
|
]
|
||||||
|
|
||||||
|
def getIntervalsList(self) -> typing.List[typing.Tuple[datetime.datetime, datetime.datetime]]:
|
||||||
|
intervals: typing.List[typing.Tuple[datetime.datetime, datetime.datetime]] = []
|
||||||
|
# Convert start and end dates to datetime objects from date objects
|
||||||
|
start = datetime.datetime.combine(self.startingDate(), datetime.time.min)
|
||||||
|
to = datetime.datetime.combine(self.endingDate(), datetime.time.max)
|
||||||
|
while start < to:
|
||||||
|
if self.interval.value == 'hour':
|
||||||
|
intervals.append((start, start + datetime.timedelta(hours=1)))
|
||||||
|
start += datetime.timedelta(hours=1)
|
||||||
|
elif self.interval.value == 'day':
|
||||||
|
intervals.append((start, start + datetime.timedelta(days=1)))
|
||||||
|
start += datetime.timedelta(days=1)
|
||||||
|
elif self.interval.value == 'week':
|
||||||
|
intervals.append((start, start + datetime.timedelta(days=7)))
|
||||||
|
start += datetime.timedelta(days=7)
|
||||||
|
elif self.interval.value == 'month':
|
||||||
|
next = (start + datetime.timedelta(days=32)).replace(day=1)
|
||||||
|
intervals.append((start, next))
|
||||||
|
start = next
|
||||||
|
|
||||||
|
logger.info('Intervals: {0}'.format(intervals))
|
||||||
|
return intervals
|
||||||
|
|
||||||
|
|
||||||
|
def adjustDate(self, d: datetime.date, isEndingDate: bool) -> datetime.date:
|
||||||
|
if self.interval.value in ('hour', 'day'):
|
||||||
|
return d
|
||||||
|
elif self.interval.value == 'week':
|
||||||
|
return (d - datetime.timedelta(days=d.weekday())).replace()
|
||||||
|
elif self.interval.value == 'month':
|
||||||
|
if not isEndingDate:
|
||||||
|
return d.replace(day=1)
|
||||||
|
else:
|
||||||
|
return (d + datetime.timedelta(days=32)).replace(day=1) - datetime.timedelta(days=1)
|
||||||
|
else:
|
||||||
|
return d
|
||||||
|
|
||||||
|
def formatDatetimeAsString(self, d: datetime.date) -> str:
|
||||||
|
if self.interval.value in ('hour', 'day'):
|
||||||
|
return d.strftime('%Y-%b-%d %H:%M:%S')
|
||||||
|
elif self.interval.value == 'week':
|
||||||
|
return d.strftime('%Y-%b-%d')
|
||||||
|
elif self.interval.value == 'month':
|
||||||
|
return d.strftime('%Y-%b')
|
||||||
|
else:
|
||||||
|
return d.strftime('%Y-%b-%d %H:%M:%S')
|
||||||
|
|
||||||
|
def startingDate(self) -> datetime.date:
|
||||||
|
return self.adjustDate(self.date_start.date(), False)
|
||||||
|
|
||||||
|
def endingDate(self) -> datetime.date:
|
||||||
|
return self.adjustDate(self.date_end.date(), True)
|
||||||
|
@ -30,6 +30,7 @@
|
|||||||
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
.. moduleauthor:: Adolfo Gómez, dkmaster at dkmon dot com
|
||||||
"""
|
"""
|
||||||
import logging
|
import logging
|
||||||
|
import datetime
|
||||||
import typing
|
import typing
|
||||||
|
|
||||||
from django.utils.translation import gettext, gettext_lazy as _
|
from django.utils.translation import gettext, gettext_lazy as _
|
||||||
@ -46,7 +47,7 @@ if typing.TYPE_CHECKING:
|
|||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
MAX_ELEMENTS = 10000
|
MAX_ELEMENTS = 10000
|
||||||
|
BIG_INTERVAL = 3600 * 24 * 30 * 12 # 12 months
|
||||||
|
|
||||||
class AuthenticatorsStats(StatsReportAuto):
|
class AuthenticatorsStats(StatsReportAuto):
|
||||||
dates = 'range'
|
dates = 'range'
|
||||||
@ -62,9 +63,6 @@ class AuthenticatorsStats(StatsReportAuto):
|
|||||||
uuid = 'a5a43bc0-d543-11ea-af8f-af01fa65994e'
|
uuid = 'a5a43bc0-d543-11ea-af8f-af01fa65994e'
|
||||||
|
|
||||||
def generate(self) -> typing.Any:
|
def generate(self) -> typing.Any:
|
||||||
since = self.date_start.date()
|
|
||||||
to = self.date_end.date()
|
|
||||||
interval = self.getIntervalInHours() * 3600
|
|
||||||
|
|
||||||
stats = []
|
stats = []
|
||||||
for a in self.getModelItems():
|
for a in self.getModelItems():
|
||||||
@ -73,69 +71,55 @@ class AuthenticatorsStats(StatsReportAuto):
|
|||||||
|
|
||||||
services = 0
|
services = 0
|
||||||
userServices = 0
|
userServices = 0
|
||||||
servicesCounterIter = iter(
|
for i in self.getIntervalsList():
|
||||||
counters.getCounters(
|
start = i[0]
|
||||||
|
end = i[1]
|
||||||
|
data = [0, 0, 0]
|
||||||
|
# Get stats for interval
|
||||||
|
for counter in counters.getCounters(
|
||||||
typing.cast('models.Authenticator', a),
|
typing.cast('models.Authenticator', a),
|
||||||
counters.CT_AUTH_SERVICES,
|
counters.CT_AUTH_SERVICES,
|
||||||
since=since,
|
since=start,
|
||||||
to=to,
|
to=end,
|
||||||
interval=interval,
|
interval=BIG_INTERVAL,
|
||||||
limit=MAX_ELEMENTS,
|
|
||||||
use_max=True,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
usersWithServicesCounterIter = iter(
|
|
||||||
counters.getCounters(
|
|
||||||
typing.cast('models.Authenticator', a),
|
|
||||||
counters.CT_AUTH_USERS_WITH_SERVICES,
|
|
||||||
since=since,
|
|
||||||
to=to,
|
|
||||||
interval=interval,
|
|
||||||
limit=MAX_ELEMENTS,
|
|
||||||
use_max=True,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
for userCounter in counters.getCounters(
|
|
||||||
typing.cast('models.Authenticator', a),
|
|
||||||
counters.CT_AUTH_USERS,
|
|
||||||
since=since,
|
|
||||||
to=to,
|
|
||||||
interval=interval,
|
|
||||||
limit=MAX_ELEMENTS,
|
limit=MAX_ELEMENTS,
|
||||||
use_max=True,
|
use_max=True,
|
||||||
):
|
):
|
||||||
try:
|
data[0] += counter[1]
|
||||||
while True:
|
|
||||||
servicesCounter = next(servicesCounterIter)
|
|
||||||
if servicesCounter[0] >= userCounter[0]:
|
|
||||||
break
|
|
||||||
if userCounter[0] == servicesCounter[0]:
|
|
||||||
services = servicesCounter[1]
|
|
||||||
except StopIteration:
|
|
||||||
pass
|
|
||||||
|
|
||||||
try:
|
for counter in counters.getCounters(
|
||||||
while True:
|
typing.cast('models.Authenticator', a),
|
||||||
uservicesCounter = next(usersWithServicesCounterIter)
|
counters.CT_AUTH_USERS_WITH_SERVICES,
|
||||||
if uservicesCounter[0] >= userCounter[0]:
|
since=start,
|
||||||
break
|
to=end,
|
||||||
if userCounter[0] == uservicesCounter[0]:
|
interval=BIG_INTERVAL,
|
||||||
userServices = uservicesCounter[1]
|
limit=MAX_ELEMENTS,
|
||||||
except StopIteration:
|
use_max=True,
|
||||||
pass
|
):
|
||||||
|
data[1] += counter[1]
|
||||||
|
for counter in counters.getCounters(
|
||||||
|
typing.cast('models.Authenticator', a),
|
||||||
|
counters.CT_AUTH_USERS,
|
||||||
|
since=start,
|
||||||
|
to=end,
|
||||||
|
interval=BIG_INTERVAL,
|
||||||
|
limit=MAX_ELEMENTS,
|
||||||
|
use_max=True,
|
||||||
|
):
|
||||||
|
data[2] += counter[1]
|
||||||
|
|
||||||
stats.append(
|
stats.append(
|
||||||
{
|
{
|
||||||
'date': userCounter[0],
|
'date': self.formatDatetimeAsString(start),
|
||||||
'users': userCounter[1] or 0,
|
'users': data[0],
|
||||||
'services': services,
|
'services': data[1],
|
||||||
'user_services': userServices,
|
'user_services': data[2],
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
logger.debug('Report Data Done')
|
logger.debug('Report Data Done')
|
||||||
return self.templateAsPDF(
|
return self.templateAsPDF(
|
||||||
'uds/reports/stats/authenticator_stats.html',
|
'uds/reports/stats/authenticator_stats.html',
|
||||||
dct={'data': stats},
|
dct={'data': stats},
|
||||||
header=gettext('Users usage list'),
|
header=ugettext('Users usage list'),
|
||||||
water=gettext('UDS Report of users usage'),
|
water=ugettext('UDS Report of users usage'),
|
||||||
)
|
)
|
||||||
|
@ -110,40 +110,23 @@ class PoolPerformanceReport(StatsReport):
|
|||||||
) -> typing.Tuple[str, typing.List, typing.List]: # pylint: disable=too-many-locals
|
) -> typing.Tuple[str, typing.List, typing.List]: # pylint: disable=too-many-locals
|
||||||
start = self.startDate.stamp()
|
start = self.startDate.stamp()
|
||||||
end = self.endDate.stamp()
|
end = self.endDate.stamp()
|
||||||
|
|
||||||
if self.samplingPoints.num() < 2:
|
|
||||||
self.samplingPoints.value = (
|
|
||||||
self.endDate.date() - self.startDate.date()
|
|
||||||
).days
|
|
||||||
if self.samplingPoints.num() < 2:
|
if self.samplingPoints.num() < 2:
|
||||||
self.samplingPoints.value = 2
|
self.samplingPoints.value = 2
|
||||||
if self.samplingPoints.num() > 32:
|
if self.samplingPoints.num() > 128:
|
||||||
self.samplingPoints.value = 32
|
self.samplingPoints.value = 128
|
||||||
|
|
||||||
samplingPoints = self.samplingPoints.num()
|
samplingPoints = self.samplingPoints.num()
|
||||||
|
|
||||||
pools = self.getPools()
|
|
||||||
|
|
||||||
if not pools:
|
|
||||||
raise Exception(_('Select at least a service pool for the report'))
|
|
||||||
|
|
||||||
logger.debug('Pools: %s', pools)
|
|
||||||
|
|
||||||
# x axis label format
|
# x axis label format
|
||||||
if end - start > 3600 * 24 * 2:
|
if end - start > 3600 * 24 * 2:
|
||||||
xLabelFormat = 'SHORT_DATE_FORMAT'
|
xLabelFormat = 'SHORT_DATE_FORMAT'
|
||||||
else:
|
else:
|
||||||
xLabelFormat = 'SHORT_DATETIME_FORMAT'
|
xLabelFormat = 'SHORT_DATETIME_FORMAT'
|
||||||
|
|
||||||
# Generate samplings interval
|
|
||||||
samplingIntervals: typing.List[typing.Tuple[int, int]] = []
|
samplingIntervals: typing.List[typing.Tuple[int, int]] = []
|
||||||
prevVal = None
|
samplingIntervalSeconds = (end - start) / samplingPoints
|
||||||
for val in range(start, end, int((end - start) / (samplingPoints + 1))):
|
for i in range(samplingPoints):
|
||||||
if prevVal is None:
|
samplingIntervals.append((int(start + i * samplingIntervalSeconds), int(start + (i + 1) * samplingIntervalSeconds)))
|
||||||
prevVal = val
|
|
||||||
continue
|
|
||||||
samplingIntervals.append((prevVal, val))
|
|
||||||
prevVal = val
|
|
||||||
|
|
||||||
# Store dataUsers for all pools
|
# Store dataUsers for all pools
|
||||||
poolsData = []
|
poolsData = []
|
||||||
@ -151,11 +134,11 @@ class PoolPerformanceReport(StatsReport):
|
|||||||
fld = StatsManager.manager().getEventFldFor('username')
|
fld = StatsManager.manager().getEventFldFor('username')
|
||||||
|
|
||||||
reportData = []
|
reportData = []
|
||||||
for p in pools:
|
for p in self.getPools():
|
||||||
dataUsers = []
|
dataUsers = []
|
||||||
dataAccesses = []
|
dataAccesses = []
|
||||||
for interval in samplingIntervals:
|
for interval in samplingIntervals:
|
||||||
key = (interval[0] + interval[1]) / 2
|
key = (interval[0] + interval[1]) // 2
|
||||||
q = (
|
q = (
|
||||||
StatsManager.manager()
|
StatsManager.manager()
|
||||||
.getEvents(
|
.getEvents(
|
||||||
@ -177,9 +160,9 @@ class PoolPerformanceReport(StatsReport):
|
|||||||
reportData.append(
|
reportData.append(
|
||||||
{
|
{
|
||||||
'name': p[1],
|
'name': p[1],
|
||||||
'date': tools.timestampAsStr(interval[0], xLabelFormat)
|
'date': tools.timestampAsStr(interval[0], 'SHORT_DATETIME_FORMAT')
|
||||||
+ ' - '
|
+ ' - '
|
||||||
+ tools.timestampAsStr(interval[1], xLabelFormat),
|
+ tools.timestampAsStr(interval[1], 'SHORT_DATETIME_FORMAT'),
|
||||||
'users': len(q),
|
'users': len(q),
|
||||||
'accesses': accesses,
|
'accesses': accesses,
|
||||||
}
|
}
|
||||||
|
@ -85,7 +85,6 @@ class CountersPoolAssigned(StatsReport):
|
|||||||
def getData(self) -> typing.List[typing.Dict[str, typing.Any]]:
|
def getData(self) -> typing.List[typing.Dict[str, typing.Any]]:
|
||||||
# Generate the sampling intervals and get dataUsers from db
|
# Generate the sampling intervals and get dataUsers from db
|
||||||
start = self.startDate.date()
|
start = self.startDate.date()
|
||||||
end = self.startDate.date() + datetime.timedelta(days=1)
|
|
||||||
|
|
||||||
data = []
|
data = []
|
||||||
|
|
||||||
@ -102,15 +101,14 @@ class CountersPoolAssigned(StatsReport):
|
|||||||
pool,
|
pool,
|
||||||
counters.CT_ASSIGNED,
|
counters.CT_ASSIGNED,
|
||||||
since=start,
|
since=start,
|
||||||
to=end,
|
to=start+datetime.timedelta(days=1),
|
||||||
max_intervals=24,
|
intervals=3600,
|
||||||
use_max=True,
|
use_max=True,
|
||||||
all=False,
|
all=False,
|
||||||
):
|
):
|
||||||
hour = x[0].hour
|
hour = x[0].hour
|
||||||
val = int(x[1])
|
val = int(x[1])
|
||||||
if hours[hour] < val:
|
hours[hour] = max(hours[hour], val)
|
||||||
hours[hour] = val
|
|
||||||
|
|
||||||
data.append({'uuid': pool.uuid, 'name': pool.name, 'hours': hours})
|
data.append({'uuid': pool.uuid, 'name': pool.name, 'hours': hours})
|
||||||
|
|
||||||
|
@ -96,10 +96,6 @@ class StatsReportLogin(StatsReport):
|
|||||||
def getRangeData(self) -> typing.Tuple[str, typing.List, typing.List]:
|
def getRangeData(self) -> typing.Tuple[str, typing.List, typing.List]:
|
||||||
start = self.startDate.stamp()
|
start = self.startDate.stamp()
|
||||||
end = self.endDate.stamp()
|
end = self.endDate.stamp()
|
||||||
# if self.samplingPoints.num() < 8:
|
|
||||||
# self.samplingPoints.value = (
|
|
||||||
# self.endDate.date() - self.startDate.date()
|
|
||||||
# ).days
|
|
||||||
if self.samplingPoints.num() < 2:
|
if self.samplingPoints.num() < 2:
|
||||||
self.samplingPoints.value = 2
|
self.samplingPoints.value = 2
|
||||||
if self.samplingPoints.num() > 128:
|
if self.samplingPoints.num() > 128:
|
||||||
@ -121,7 +117,7 @@ class StatsReportLogin(StatsReport):
|
|||||||
data = []
|
data = []
|
||||||
reportData = []
|
reportData = []
|
||||||
for interval in samplingIntervals:
|
for interval in samplingIntervals:
|
||||||
key = (interval[0] + interval[1]) / 2
|
key = (interval[0] + interval[1]) // 2
|
||||||
val = (
|
val = (
|
||||||
StatsManager.manager()
|
StatsManager.manager()
|
||||||
.getEvents(
|
.getEvents(
|
||||||
@ -132,12 +128,12 @@ class StatsReportLogin(StatsReport):
|
|||||||
)
|
)
|
||||||
.count()
|
.count()
|
||||||
)
|
)
|
||||||
data.append((key, val)) # @UndefinedVariable
|
data.append((key, val))
|
||||||
reportData.append(
|
reportData.append(
|
||||||
{
|
{
|
||||||
'date': tools.timestampAsStr(interval[0], xLabelFormat)
|
'date': tools.timestampAsStr(interval[0], 'SHORT_DATETIME_FORMAT')
|
||||||
+ ' - '
|
+ ' - '
|
||||||
+ tools.timestampAsStr(interval[1], xLabelFormat),
|
+ tools.timestampAsStr(interval[1], 'SHORT_DATETIME_FORMAT'),
|
||||||
'users': val,
|
'users': val,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
@ -163,13 +159,6 @@ class StatsReportLogin(StatsReport):
|
|||||||
return dataWeek, dataHour, dataWeekHour
|
return dataWeek, dataHour, dataWeekHour
|
||||||
|
|
||||||
def generate(self):
|
def generate(self):
|
||||||
# Sample query:
|
|
||||||
# 'SELECT *, count(*) as number, CEIL(stamp/(3600))*3600 as block'
|
|
||||||
# ' FROM {table}'
|
|
||||||
# ' WHERE event_type = 0 and stamp >= {start} and stamp <= {end}'
|
|
||||||
# ' GROUP BY CEIL(stamp/(3600))'
|
|
||||||
# ' ORDER BY block'
|
|
||||||
|
|
||||||
xLabelFormat, data, reportData = self.getRangeData()
|
xLabelFormat, data, reportData = self.getRangeData()
|
||||||
|
|
||||||
#
|
#
|
||||||
|
Loading…
Reference in New Issue
Block a user