* Remove six for PY2 compatability since our dependencies are not, PY2 is legacy. * Removed usages of utils from future/past libraries since they are deprecated. This includes 'from __future__ ...' and 'from past...' statements. * Removed compatibility imports for PY2, switched from six imports to standard library imports. * Removed utils code blocks that handle operations depending on PY2/3 versions. * Removed 'from __future__ ...' lines from templates/code generators * Used PY3 syntaxes in place of PY2 compatible blocks. eg: metaclass
150 lines
4.3 KiB
Python
150 lines
4.3 KiB
Python
# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
|
# MIT License. See license.txt
|
|
|
|
import frappe
|
|
import frappe.defaults
|
|
import datetime
|
|
from frappe.utils import get_datetime, add_to_date, getdate
|
|
from frappe.utils.data import get_first_day, get_first_day_of_week, get_quarter_start, get_year_start,\
|
|
get_last_day, get_last_day_of_week, get_quarter_ending, get_year_ending
|
|
|
|
# global values -- used for caching
|
|
dateformats = {
|
|
'yyyy-mm-dd': '%Y-%m-%d',
|
|
'mm/dd/yyyy': '%m/%d/%Y',
|
|
'mm-dd-yyyy': '%m-%d-%Y',
|
|
"mm/dd/yy": "%m/%d/%y",
|
|
'dd-mmm-yyyy': '%d-%b-%Y', # numbers app format
|
|
'dd/mm/yyyy': '%d/%m/%Y',
|
|
'dd.mm.yyyy': '%d.%m.%Y',
|
|
'dd-mm-yyyy': '%d-%m-%Y',
|
|
"dd/mm/yy": "%d/%m/%y",
|
|
}
|
|
|
|
def user_to_str(date, date_format=None):
|
|
if not date: return date
|
|
|
|
if not date_format:
|
|
date_format = get_user_date_format()
|
|
|
|
try:
|
|
return datetime.datetime.strptime(date,
|
|
dateformats[date_format]).strftime('%Y-%m-%d')
|
|
except ValueError:
|
|
raise ValueError("Date %s must be in format %s" % (date, date_format))
|
|
|
|
def parse_date(date):
|
|
"""tries to parse given date to system's format i.e. yyyy-mm-dd. returns a string"""
|
|
parsed_date = None
|
|
|
|
if " " in date:
|
|
# as date-timestamp, remove the time part
|
|
date = date.split(" ")[0]
|
|
|
|
# why the sorting? checking should be done in a predictable order
|
|
check_formats = [None] + sorted(list(dateformats),
|
|
reverse=not get_user_date_format().startswith("dd"))
|
|
|
|
for f in check_formats:
|
|
try:
|
|
parsed_date = user_to_str(date, f)
|
|
if parsed_date:
|
|
break
|
|
except ValueError:
|
|
pass
|
|
|
|
if not parsed_date:
|
|
raise Exception("""Cannot understand date - '%s'.
|
|
Try formatting it like your default format - '%s'""" % (date, get_user_date_format())
|
|
)
|
|
|
|
return parsed_date
|
|
|
|
def get_user_date_format():
|
|
if getattr(frappe.local, "user_date_format", None) is None:
|
|
frappe.local.user_date_format = frappe.defaults.get_global_default("date_format") or "yyyy-mm-dd"
|
|
|
|
return frappe.local.user_date_format
|
|
|
|
def datetime_in_user_format(date_time):
|
|
if not date_time:
|
|
return ""
|
|
if isinstance(date_time, str):
|
|
date_time = get_datetime(date_time)
|
|
from frappe.utils import formatdate
|
|
return formatdate(date_time.date()) + " " + date_time.strftime("%H:%M")
|
|
|
|
def get_dates_from_timegrain(from_date, to_date, timegrain="Daily"):
|
|
from_date = getdate(from_date)
|
|
to_date = getdate(to_date)
|
|
|
|
days = months = years = 0
|
|
if "Daily" == timegrain:
|
|
days = 1
|
|
elif "Weekly" == timegrain:
|
|
days = 7
|
|
elif "Monthly" == timegrain:
|
|
months = 1
|
|
elif "Quarterly" == timegrain:
|
|
months = 3
|
|
|
|
if "Weekly" == timegrain:
|
|
dates = [get_last_day_of_week(from_date)]
|
|
else:
|
|
dates = [get_period_ending(from_date, timegrain)]
|
|
|
|
while getdate(dates[-1]) < getdate(to_date):
|
|
if "Weekly" == timegrain:
|
|
date = get_last_day_of_week(add_to_date(dates[-1], years=years, months=months, days=days))
|
|
else:
|
|
date = get_period_ending(add_to_date(dates[-1], years=years, months=months, days=days), timegrain)
|
|
dates.append(date)
|
|
return dates
|
|
|
|
def get_from_date_from_timespan(to_date, timespan):
|
|
days = months = years = 0
|
|
if timespan == "Last Week":
|
|
days = -7
|
|
if timespan == "Last Month":
|
|
months = -1
|
|
elif timespan == "Last Quarter":
|
|
months = -3
|
|
elif timespan == "Last Year":
|
|
years = -1
|
|
elif timespan == "All Time":
|
|
years = -50
|
|
return add_to_date(to_date, years=years, months=months, days=days,
|
|
as_datetime=True)
|
|
|
|
def get_period(date, interval='Monthly'):
|
|
date = getdate(date)
|
|
months = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
|
|
return {
|
|
'Daily': date.strftime('%d-%m-%y'),
|
|
'Weekly': date.strftime('%d-%m-%y'),
|
|
'Monthly': str(months[date.month - 1]) + ' ' + str(date.year),
|
|
'Quarterly': 'Quarter ' + str(((date.month-1)//3)+1) + ' ' + str(date.year),
|
|
'Yearly': str(date.year)
|
|
}[interval]
|
|
|
|
def get_period_beginning(date, timegrain, as_str=True):
|
|
return getdate({
|
|
'Daily': date,
|
|
'Weekly': get_first_day_of_week(date),
|
|
'Monthly': get_first_day(date),
|
|
'Quarterly': get_quarter_start(date),
|
|
'Yearly': get_year_start(date)
|
|
}[timegrain])
|
|
|
|
def get_period_ending(date, timegrain):
|
|
date = getdate(date)
|
|
if timegrain == 'Daily':
|
|
return date
|
|
else:
|
|
return getdate({
|
|
'Daily': date,
|
|
'Weekly': get_last_day_of_week(date),
|
|
'Monthly': get_last_day(date),
|
|
'Quarterly': get_quarter_ending(date),
|
|
'Yearly': get_year_ending(date)
|
|
}[timegrain])
|