110 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			110 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # The Mia! Accounting Project.
 | |
| # Author: imacat@mail.imacat.idv.tw (imacat), 2023/3/7
 | |
| 
 | |
| #  Copyright (c) 2023 imacat.
 | |
| #
 | |
| #  Licensed under the Apache License, Version 2.0 (the "License");
 | |
| #  you may not use this file except in compliance with the License.
 | |
| #  You may obtain a copy of the License at
 | |
| #
 | |
| #      http://www.apache.org/licenses/LICENSE-2.0
 | |
| #
 | |
| #  Unless required by applicable law or agreed to in writing, software
 | |
| #  distributed under the License is distributed on an "AS IS" BASIS,
 | |
| #  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| #  See the License for the specific language governing permissions and
 | |
| #  limitations under the License.
 | |
| """The utilities to export the report as CSV for download.
 | |
| 
 | |
| """
 | |
| import csv
 | |
| from abc import ABC, abstractmethod
 | |
| from datetime import timedelta, date
 | |
| from decimal import Decimal
 | |
| from io import StringIO
 | |
| from urllib.parse import quote
 | |
| 
 | |
| from flask import Response
 | |
| 
 | |
| from accounting.report.period import Period
 | |
| 
 | |
| 
 | |
| class BaseCSVRow(ABC):
 | |
|     """The base CSV row."""
 | |
| 
 | |
|     @property
 | |
|     @abstractmethod
 | |
|     def values(self) -> list[str | Decimal | None]:
 | |
|         """Returns the values of the row.
 | |
| 
 | |
|         :return: The values of the row.
 | |
|         """
 | |
| 
 | |
| 
 | |
| def csv_download(filename: str, rows: list[BaseCSVRow]) -> Response:
 | |
|     """Exports the data rows as a CSV file for download.
 | |
| 
 | |
|     :param filename: The download file name.
 | |
|     :param rows: The data rows.
 | |
|     :return: The response for download the CSV file.
 | |
|     """
 | |
|     with StringIO() as fp:
 | |
|         writer = csv.writer(fp)
 | |
|         writer.writerows([x.values for x in rows])
 | |
|         fp.seek(0)
 | |
|         response: Response = Response(fp.read(), mimetype="text/csv")
 | |
|         response.headers["Content-Disposition"] \
 | |
|             = f"attachment; filename={quote(filename)}"
 | |
|         return response
 | |
| 
 | |
| 
 | |
| def period_spec(period: Period) -> str:
 | |
|     """Constructs the period specification to be used in the filename.
 | |
| 
 | |
|     :param period: The period.
 | |
|     :return: The period specification to be used in the filename.
 | |
|     """
 | |
|     start: str | None = __get_start_str(period.start)
 | |
|     end: str | None = __get_end_str(period.end)
 | |
|     if period.start is None and period.end is None:
 | |
|         return "all-time"
 | |
|     if start == end:
 | |
|         return start
 | |
|     if period.start is None:
 | |
|         return f"until-{end}"
 | |
|     if period.end is None:
 | |
|         return f"since-{start}"
 | |
|     return f"{start}-{end}"
 | |
| 
 | |
| 
 | |
| def __get_start_str(start: date | None) -> str | None:
 | |
|     """Returns the string representation of the start date.
 | |
| 
 | |
|     :param start: The start date.
 | |
|     :return: The string representation of the start date, or None if the start
 | |
|         date is None.
 | |
|     """
 | |
|     if start is None:
 | |
|         return None
 | |
|     if start.month == 1 and start.day == 1:
 | |
|         return str(start.year)
 | |
|     if start.day == 1:
 | |
|         return start.strftime("%Y%m")
 | |
|     return start.strftime("%Y%m%d")
 | |
| 
 | |
| 
 | |
| def __get_end_str(end: date | None) -> str | None:
 | |
|     """Returns the string representation of the end date.
 | |
| 
 | |
|     :param end: The end date.
 | |
|     :return: The string representation of the end date, or None if the end
 | |
|         date is None.
 | |
|     """
 | |
|     if end is None:
 | |
|         return None
 | |
|     if end.month == 12 and end.day == 31:
 | |
|         return str(end.year)
 | |
|     if (end + timedelta(days=1)).day == 1:
 | |
|         return end.strftime("%Y%m")
 | |
|     return end.strftime("%Y%m%d")
 |