diff --git a/projects/data_cleaning/Group_discussion.md b/projects/data_cleaning/Group_discussion.md
new file mode 100644
index 0000000..a05b5b0
--- /dev/null
+++ b/projects/data_cleaning/Group_discussion.md
@@ -0,0 +1,30 @@
+# On-line Group Discussion
+
+Related to the time series cleaning project.
+
+## Module 1
+
+- Study this script `clean_project_data_v4_final2.py` for 3 minutes
+- Consider what you could do to improve it
+- Q1: Discuss in your group how to improve the script.
+- Q2: Version control. What is your experience with version control?
+ - Think about a project you've worked on in the past that involved collaborating with others on code. What challenges did you face, and how do you think Git and GitHub could have helped to address those challenges?
+
+
+## Module 2
+
+- Q1: In your course project homework, you refactored the script to use functions. How did it go?
+- Q2: Classes. If you should introduce classes to improve the code, which classes should it be and why?
+- Q3: [Optional] What are some problems with poorly designed code (based on your own experience or from the book)?
+
+
+## Module 3
+
+- Q1: In your course project homework from last module, you implemented modules and classes, how did it go? Any reflections?
+- Q2: What is you past experience with testing code?
+- Q3: Things can go wrong when executing your code, how should you handle that? Check inputs? try-catch statements? What are pros and cons between different approaches?
+
+
+## Module 4
+
+- Q2:
diff --git a/projects/data_cleaning/Project_module_01.md b/projects/data_cleaning/Project_module_01.md
new file mode 100644
index 0000000..b89df3d
--- /dev/null
+++ b/projects/data_cleaning/Project_module_01.md
@@ -0,0 +1,25 @@
+## Module 1: GitHub and basic functions
+
+- 1.1 GitHub repo
+ - 1.1.1 Create a new GitHub repository "timeseriescleaner"
+ - private, no template, add readme, gitignore python, no license
+ - 1.1.2 Go to repo settings/Collaborators add your instructors and your "buddy"
+ - 1.1.3 Clone repo to local machine
+ - [Optional] Create virtual environment for this course project (use venv or mamba/conda environment)
+ - 1.1.4 Download the provided Python script and add it to the repo
+ - 1.1.5 Commit the file and push the changes (Check that the file can be found on GitHub)
+ - 1.1.6 Open the project in vscode and make a single character change to the file (add a comment)
+ - 1.1.7 Commit the changes (Check that it works on GitHub)
+- 1.2 Functions
+ - 1.2.1 Create a local branch "refactor-functions"
+ - 1.2.2 Refactor the code to use functions (`clean_spikes`, `clean_outofrange`, `clean_flat`, `plot_timeseries`)
+ - for data in [data1, data2, data3]:
+ - data_original = data.copy()
+ - data = clean_spikes(data, max_jump=10)
+ - data = clean_outofrange(data, min_val=0, max_val=50)
+ - data = clean_flat(data, flat_period=5)
+ - plot_timeseries(data_original, data)
+ - 1.2.3 Check that your code and produce the same results as before (you should not change the functionality!)
+ - 1.2.4 Commit your code in 1 or more commits (in the end your code should be approximately 75 lines long)
+- Create a pull request in GitHub and "request review" from your reviewers
+- Wait for feedback, Adjust code until approval, then merge (and delete branch)
\ No newline at end of file
diff --git a/projects/data_cleaning/Project_module_02.md b/projects/data_cleaning/Project_module_02.md
new file mode 100644
index 0000000..a9885cd
--- /dev/null
+++ b/projects/data_cleaning/Project_module_02.md
@@ -0,0 +1,29 @@
+## Module 2: Modules and classes
+
+- Create new branch "modules-classes" (Make sure changes from last module have been merged, and that you start from the main branch)
+- 2.1 Function arguments
+ - Add default arguments to the functions. Commit.
+ - Make sure that you only use positional arguments where there is only one argument. Use keyword arguments everywhere else. Commit.
+- 2.2 Modules
+ - Move cleaner functions into a separate module "cleaning.py". Commit.
+ - Move the plotting function into a separate module "plotting.py". Commit.
+ - Rename the script `main.py` and execute the cleaning and plotting.
+ - from cleaning import ...
+ - from plotting import ...
+ - Check that it runs!
+- 2.3 Classes
+ - Organize the cleaning functions into classes that all have the same structure (an init method and a clean method)
+ - SpikeCleaner
+ - `def __init__(max_jump)`
+ - `def clean(data)`
+ - modify `main.py` and check that it runs
+ - cleaners = [
+ - SpikeCleaner(max_jump=10),
+ - OutOfRangeCleaner(min_val=0, max_val=50),
+ - FlatPeriodCleaner(flat_period=5),
+ - ]
+ - for cleaner in cleaners:
+ - data = cleaner.clean(data)
+ - Download notebook_A and csv file and make sure it runs. (remove any remaining print statements)
+- Create pull request in GitHub and "request review" from your reviewers
+- Get feedback, Adjust code until approval, then merge (and delete branch)
\ No newline at end of file
diff --git a/projects/data_cleaning/Project_module_03.md b/projects/data_cleaning/Project_module_03.md
new file mode 100644
index 0000000..52b5eea
--- /dev/null
+++ b/projects/data_cleaning/Project_module_03.md
@@ -0,0 +1,35 @@
+## Module 3: Installable package and pytest
+
+- Create new branch "package-test" (Make sure changes from last module have been merged, and that you start from the main branch)
+- Make sure pytest and pytest-cov are installed
+- 3.1 Installable package
+ - 3.1.1 Organize the files into folders and add setup.py. Call your package tscleaner.
+ - subfolders: tscleaner, scripts, notebooks, tests
+ - make init-file in tscleaner with
+ - `from .cleaning import SpikeCleaner, FlatPeriodCleaner, OutOfRangeCleaner`
+ - `from .plotting import plot_timeseries`
+ - create a setup.py in the root with the following content (change with your data):
+ - from setuptools import setup, find_packages
+ - setup(
+ name='MyPackageName',
+ version='0.0.1',
+ url='https://github.com/mypackage.git',
+ author='Author Name',
+ author_email='author@gmail.com',
+ description='Description of my package',
+ packages=find_packages(),
+ install_requires=['numpy', 'matplotlib'],
+ )
+ - 3.1.2 Install the package in editable mode.
+ - `>pip install -e .`
+ - 3.1.3 Modify import statements in notebook_A and script main.py and make sure they run.
+ - 3.1.4 Modify cleaner tools by raising exceptions for invalid inputs.
+ - 3.1.5 Move the csv file to `/tests/testdata` and update notebook with relative path to the file
+- 3.2 Pytest
+ - 3.2.1 Write unit tests with pytest in the `/tests` folder. Create an empty init-py file in the folder. Create a file `test_cleaning.py` and create at least five tests that verify that the cleaning tools work as intended
+ - [Optional] Consider to make a fixture that reads the csv file and you can read in all tests
+ - 3.2.2 Run the tests from the commandline by writting `>pytest` in the project root (can you also run the tests from VSCode?)
+ - 3.2.3 Assess the test coverage with `>pytest --cov=tscleaner tests`
+ - Optional: Get coverage as html with `>pytest --cov=tscleaner --cov-report html` (check the index.html in the htmlcov subfolder afterwards)
+- Create pull request in GitHub and "request review" from your reviewers
+- Get feedback, Adjust code until approval, then merge (and delete branch)
\ No newline at end of file
diff --git a/projects/data_cleaning/Project_module_04.md b/projects/data_cleaning/Project_module_04.md
new file mode 100644
index 0000000..b4b2781
--- /dev/null
+++ b/projects/data_cleaning/Project_module_04.md
@@ -0,0 +1,28 @@
+## Module 4: GitHub actions and auto-formatting
+
+- Create new branch "action-formatting" (Make sure changes from last module have been merged, and that you start from the main branch)
+- 4.1 Github Action
+ - 4.1.1 Copy the GitHub action "python-app.yml" from the python template https://github.com/DHI/template-python-library to your own library (make sure it sits in the same folder).
+ - 4.1.2 Change all occurrences of "my_library" in the yml file to your package name "tscleaner"
+ - 4.1.3 Comment out the line with "ruff-action" with "#"
+ - 4.1.4 Commit, push and create a pull request; the tests should now run, verify that they all run before you move on
+- 4.2 Ruff
+ - 4.2.1 Enable the "ruff-action" be removing the "#" you added above
+ - 4.2.2 Commit and push, your actions will probably fail now - inspect the problems by clicking the red cross (did you also get an email?)
+ - 4.2.3 Install "ruff" on your local machine with mamba/conda/pip
+ - 4.2.4 Navigate to your project root folder and run ruff with "ruff ."
+ - 4.2.5 Add `__all__ = ["SpikeCleaner", "FlatPeriodCleaner", "OutOfRangeCleaner", "plot_timeseries"]` to your `__init__.py` file and fix remaining issues until ruff passes
+ - 4.2.6 Commit, push and verify that you action now succeeds
+- 4.3 Black
+ - 4.3.1 Install "black" on your local machine with mamba/conda/pip
+ - 4.3.2 Run black from your project root folder; inspect the differences; commit
+- 4.4 pyproject.toml
+ - Copy the pyproject.toml from the python template https://github.com/DHI/template-python-library (this file will replace your setup.py)
+ - Modify to fit your package
+ - Remove the setup.py
+ - Commit, push and verify that the GitHub action runs
+ - If it fails, you probably forgot some dependencies - go back and fix
+ - [Optional] You should also re-install your local package with ">pip install --upgrade -e ."
+- 4.5 [Optional] Enable black and ruff extensions in VSCode; set black to run on save
+- Create pull request in GitHub and "request review" from your reviewers
+- Get feedback, Adjust code until approval, then merge (and delete branch)
\ No newline at end of file
diff --git a/projects/data_cleaning/Project_module_05.md b/projects/data_cleaning/Project_module_05.md
new file mode 100644
index 0000000..b1be10e
--- /dev/null
+++ b/projects/data_cleaning/Project_module_05.md
@@ -0,0 +1,21 @@
+## Module 5: Object-oriented design
+
+- Create new branch "oop-dataclasses" (Make sure changes from last module have been merged, and that you start from the main branch)
+- 5.1 Type Hints
+ - Add type hints to all functions and methods. Commit
+- 5.2 Data class
+ - Make all the cleaner classes dataclasses.
+ - remove the init method (not needed anymore)
+ - Check that the notebook still runs and that the classes indeed work as data classes (e.g. have a string representation and support equality testing etc)
+ - Commit
+- 5.3 Module level function
+ - Make a private module function _print_stats() that prints the number of cleaned values
+ - call from each of the clean methods
+- 5.4 Composition or inheritance
+ - Create a new cleaner class called CleanerWorkflow that takes a list of cleaners when constructed and has a clean method that run all the cleaners' clean methods.
+ - Modify the notebook to use the CleanerWorkflow instead of looping over the cleaners
+ - Consider what type of validation you would want CleanerWorkflow
+ - Consider whether it would be better to create a base class BaseCleaner - write down your considerations as a comment in the pull request, refer to specific lines of code
+ - e.g. how would you handle e.g. common plotting functionality in the cleaner classes?
+- Create pull request in GitHub and "request review" from your reviewers
+- Get feedback, Adjust code until approval, then merge (and delete branch)
\ No newline at end of file
diff --git a/projects/data_cleaning/Project_module_06.md b/projects/data_cleaning/Project_module_06.md
new file mode 100644
index 0000000..d54d10c
--- /dev/null
+++ b/projects/data_cleaning/Project_module_06.md
@@ -0,0 +1,16 @@
+## Module 6: Documentation
+
+- Create new branch "docs" (Make sure changes from last module have been merged, and that you start from the main branch)
+- 6.1 README
+ - Write a README file with basic information about the project.
+- 6.2 Docstrings
+ - Write NumPy style docstrings for all functions and classes.
+ - [Optional] Install the autodocstrings extension in VSCode (set the style to NumPy)
+- 6.3 mkdocs
+ - Install mkdocs, mkdocstrings and material design `mamba/pip install mkdocstrings-python mkdocs-material`
+ - Create a `mkdocs.yml` file (copy from https://github.com/DHI/template-python-library and adapt).
+ - Create a docs folder and create a markdown file `index.md` inside.
+ - Create API documentation locally using `>mkdocs serve`.
+ - Check the generated HTML documentation.
+- Create pull request in GitHub and "request review" from your reviewers
+- Get feedback, Adjust code until approval, then merge (and delete branch)
\ No newline at end of file
diff --git a/projects/data_cleaning/Project_module_07.md b/projects/data_cleaning/Project_module_07.md
new file mode 100644
index 0000000..4c54390
--- /dev/null
+++ b/projects/data_cleaning/Project_module_07.md
@@ -0,0 +1,5 @@
+## Module 7: Publishing
+- Add a license
+- Change version number to 0.1.0
+- Build the package with hatchling.
+- Publish the package to the PyPI Test Server.
\ No newline at end of file
diff --git a/projects/data_cleaning/Project_overview.md b/projects/data_cleaning/Project_overview.md
new file mode 100644
index 0000000..fb5fab5
--- /dev/null
+++ b/projects/data_cleaning/Project_overview.md
@@ -0,0 +1,49 @@
+# Course project: Time Series Data Cleaning
+
+## Module 1: GitHub and basic functions
+
+- 1.1 GitHub repo
+- 1.2 Functions
+
+## Module 2: Modules and classes
+
+- 2.1 Function arguments
+- 2.2 Modules
+- 2.3 Classes
+
+
+
+## Module 3: Installable package and pytest
+
+- 3.1 Installable package
+- 3.2 Pytest
+
+
+## Module 4: GitHub actions and auto-formatting
+
+- 4.1 Github Action
+- 4.2 Ruff
+- 4.3 Black
+- 4.4 pyproject.toml
+
+
+## Module 5: Object-oriented design
+
+- 5.1 Type Hints
+- 5.2 Data class
+- 5.3 Module level function
+- 5.4 Composition or inheritance
+
+
+## Module 6: Documentation
+
+- 6.1 README
+- 6.2 Docstrings
+- 6.3 mkdocs
+
+
+## Module 7: Publishing
+- Add a license
+- Change version number to 0.1.0
+- Build the package with hatchling.
+- Publish the package to the PyPI Test Server.
diff --git a/projects/data_cleaning/clean_project_data_v4_final2.py b/projects/data_cleaning/clean_project_data_v4_final2.py
new file mode 100644
index 0000000..0b80885
--- /dev/null
+++ b/projects/data_cleaning/clean_project_data_v4_final2.py
@@ -0,0 +1,184 @@
+import pandas as pd
+import numpy as np
+from datetime import datetime, timedelta
+import matplotlib.pyplot as plt
+
+# Create date range
+date_rng = pd.date_range(start="1/1/2020", end="1/31/2020", freq="D")
+
+# Sample time series data with DateTimeIndex
+data1 = pd.Series([1, 2, -1, 4, 5, 20, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
+ 21, 22, 24, 24, 24, 24, 24, 24, 29, 30, 31], index=date_rng)
+data2 = pd.Series([5, 6, 200, 8, 9, 10, 11, 12, 300, 14, 15, 16, 17, 18, 19, 20, 21, 22,
+ 23, 24, 25, 26, 27, 27, 27, 30, 31, 32, 33, 34, 35], index=date_rng)
+data3 = pd.Series([15, 16, 11, 18, 400, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31,
+ 32, 33, 34, 35, 36, 37, 38, 39, 45, 45, 45, 45, 45, 45], index=date_rng)
+
+
+# Cleaning data1
+print("\nCleaning data1")
+data1_original = data1.copy()
+
+# Checking for jumps
+print("Checking for jumps in data1")
+max_jump=10
+prev_value = data1.iloc[0]
+for t, value in data1.items():
+ if abs(value - prev_value) <= max_jump:
+ # "Value ok"
+ data1[t] = value
+ prev_value = value
+ else:
+ data1[t] = np.nan
+ print("Jump detected and value removed on", t, ":", value)
+print(f"Data removed: {data1_original[~data1_original.isin(data1)]}")
+# print("Data1 after jump check:", data1)
+
+# Checking for values in range
+min_val = 0
+max_val = 50
+for t, value in data1.items():
+ # print("Checking value on", t, ":", value)
+ if min_val <= value <= max_val:
+ pass
+ # print("Value ok:", value)
+ else:
+ data1[t] = np.nan
+ print("Value removed:", value)
+print(f"Data removed: {data1_original[~data1_original.isin(data1)]}")
+# print("Data1 after range check:", data1)
+
+
+# Checking for flat periods
+print("Checking for flat periods in data1")
+flat_period = 5
+i = 0
+while i < len(data1) - flat_period:
+ if len(set(data1[i: i + flat_period + 1])) == 1:
+ print("Removing flat period starting at index:", i)
+ data1[i: i + flat_period + 1] = np.nan
+ i += flat_period
+ else:
+ i += 1
+print(f"Data removed: {data1_original[~data1_original.isin(data1)]}")
+# print("Data1 after flat period check:", data1)
+
+
+# Cleaning data2
+print("\nCleaning data2")
+data2_original = data2.copy()
+
+# Checking for jumps
+print("Checking for jumps in data2")
+max_jump=10
+prev_value = data2.iloc[0]
+for t, value in data2.items():
+ if abs(value - prev_value) <= max_jump:
+ # "Value ok"
+ data2[t] = value
+ prev_value = value
+ else:
+ data2[t] = np.nan
+ print("Jump detected and value removed on", t, ":", value)
+print(f"Data removed: {data2_original[~data2_original.isin(data2)]}")
+# print("data2 after jump check:", data2)
+
+# Checking for values in range
+min_val = 0
+max_val = 50
+for t, value in data2.items():
+ # print("Checking value on", t, ":", value)
+ if min_val <= value <= max_val:
+ pass
+ # print("Value ok:", value)
+ else:
+ data2[t] = np.nan
+ print("Value removed:", value)
+print(f"Data removed: {data2_original[~data2_original.isin(data2)]}")
+# print("data2 after range check:", data2)
+
+
+# Checking for flat periods
+print("Checking for flat periods in data2")
+flat_period = 5
+i = 0
+while i < len(data2) - flat_period:
+ if len(set(data2[i: i + flat_period + 1])) == 1:
+ print("Removing flat period starting at index:", i)
+ data2[i: i + flat_period + 1] = np.nan
+ i += flat_period
+ else:
+ i += 1
+print(f"Data removed: {data2_original[~data2_original.isin(data2)]}")
+# print("data2 after flat period check:", data2)
+
+# print("Final cleaned data2:", data2)
+
+# Cleaning data3
+print("\nCleaning data3")
+data3_original = data3.copy()
+
+# Checking for jumps
+print("Checking for jumps in data3")
+max_jump=10
+prev_value = data3.iloc[0]
+for t, value in data3.items():
+ if abs(value - prev_value) <= max_jump:
+ # "Value ok"
+ data3[t] = value
+ prev_value = value
+ else:
+ data3[t] = np.nan
+ print("Jump detected and value removed on", t, ":", value)
+print(f"Data removed: {data3_original[~data3_original.isin(data3)]}")
+# print("data3 after jump check:", data3)
+
+# Checking for values in range
+min_val = 0
+max_val = 50
+for t, value in data3.items():
+ # print("Checking value on", t, ":", value)
+ if min_val <= value <= max_val:
+ pass
+ # print("Value ok:", value)
+ else:
+ data3[t] = np.nan
+ print("Value removed:", value)
+print(f"Data removed: {data3_original[~data3_original.isin(data3)]}")
+# print("data3 after range check:", data3)
+
+
+# Checking for flat periods
+print("Checking for flat periods in data3")
+flat_period = 5
+i = 0
+while i < len(data3) - flat_period:
+ if len(set(data3[i: i + flat_period + 1])) == 1:
+ print("Removing flat period starting at index:", i)
+ data3[i: i + flat_period + 1] = np.nan
+ i += flat_period
+ else:
+ i += 1
+print(f"Data removed: {data3_original[~data3_original.isin(data3)]}")
+# print("data3 after flat period check:", data3)
+
+# print("Final cleaned data3:", data3)
+
+## plot data showing outliers as red dots
+plt.figure(figsize=(10, 5))
+plt.plot(data1_original, '.', color="red")
+plt.plot(data1, '.', color="green")
+plt.title("Data1")
+plt.show()
+
+plt.figure(figsize=(10, 5))
+plt.plot(data2_original, '.', color="red")
+plt.plot(data2, '.', color="green")
+plt.title("Data2")
+plt.show()
+
+plt.figure(figsize=(10, 5))
+plt.plot(data3_original, '.', color="red")
+plt.plot(data3, '.', color="green")
+plt.title("Data3")
+plt.show()
\ No newline at end of file
diff --git a/projects/data_cleaning/example_data1.csv b/projects/data_cleaning/example_data1.csv
new file mode 100644
index 0000000..7730a4b
--- /dev/null
+++ b/projects/data_cleaning/example_data1.csv
@@ -0,0 +1,32 @@
+,series1
+2020-01-01,1.0
+2020-01-02,2.0
+2020-01-03,-1.0
+2020-01-04,4.0
+2020-01-05,5.0
+2020-01-06,20.0
+2020-01-07,7.0
+2020-01-08,8.0
+2020-01-09,9.0
+2020-01-10,10.0
+2020-01-11,11.0
+2020-01-12,12.0
+2020-01-13,13.0
+2020-01-14,14.0
+2020-01-15,15.0
+2020-01-16,16.0
+2020-01-17,17.0
+2020-01-18,18.0
+2020-01-19,19.0
+2020-01-20,20.0
+2020-01-21,21.0
+2020-01-22,22.0
+2020-01-23,24.0
+2020-01-24,24.0
+2020-01-25,24.0
+2020-01-26,24.0
+2020-01-27,24.0
+2020-01-28,24.0
+2020-01-29,29.0
+2020-01-30,30.0
+2020-01-31,31.0
diff --git a/projects/data_cleaning/notebook_A.ipynb b/projects/data_cleaning/notebook_A.ipynb
new file mode 100644
index 0000000..e038296
--- /dev/null
+++ b/projects/data_cleaning/notebook_A.ipynb
@@ -0,0 +1,365 @@
+{
+ "cells": [
+ {
+ "cell_type": "markdown",
+ "metadata": {},
+ "source": [
+ "# Clean data from a file"
+ ]
+ },
+ {
+ "cell_type": "code",
+ "execution_count": 1,
+ "metadata": {},
+ "outputs": [],
+ "source": [
+ "# useful if your change your modules after starting the kernel\n",
+ "%load_ext autoreload\n",
+ "%autoreload 2"
+ ]
+ },
+ {
+ "cell_type": "code",
+ "execution_count": 2,
+ "metadata": {},
+ "outputs": [],
+ "source": [
+ "import pandas as pd\n",
+ "from cleaning import SpikeCleaner, OutOfRangeCleaner, FlatPeriodCleaner\n",
+ "from plotting import plot_timeseries"
+ ]
+ },
+ {
+ "cell_type": "code",
+ "execution_count": 3,
+ "metadata": {},
+ "outputs": [],
+ "source": [
+ "fn = \"./example_data1.csv\"\n",
+ "df = pd.read_csv(fn, index_col=0, parse_dates=True, dtype=float)"
+ ]
+ },
+ {
+ "cell_type": "code",
+ "execution_count": 4,
+ "metadata": {},
+ "outputs": [
+ {
+ "data": {
+ "text/html": [
+ "