From 8819ab8e6503b13016584e01b9725a316b61398a Mon Sep 17 00:00:00 2001 From: Jay Date: Sat, 6 Jan 2024 01:31:43 -0500 Subject: [PATCH 01/12] Update README.md --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index abf590c..a79c613 100644 --- a/README.md +++ b/README.md @@ -1,5 +1,5 @@ -# Blackboard Scraper +# Archive-Me Archive Me is a tool for students to easily archive and track their course materials on Blackboard. It helps students store and organize their coursework for long-term access and reference. This tool is useful for college students to preserve their academic records. From 9c15375fac33dc51d58ed17c373fd85cbe2e524d Mon Sep 17 00:00:00 2001 From: Jay Date: Sun, 7 Jan 2024 15:29:45 -0500 Subject: [PATCH 02/12] Create Unit-Tests.yml --- .github/workflows/Unit-Tests.yml | 30 ++++++++++++++++++++++++++++++ 1 file changed, 30 insertions(+) create mode 100644 .github/workflows/Unit-Tests.yml diff --git a/.github/workflows/Unit-Tests.yml b/.github/workflows/Unit-Tests.yml new file mode 100644 index 0000000..b9633a1 --- /dev/null +++ b/.github/workflows/Unit-Tests.yml @@ -0,0 +1,30 @@ +name: Run Blackboard Session Tests + +on: + push: + branches: + - main + pull_request: + branches: + - main + +jobs: + test: + runs-on: ubuntu-latest + steps: + - name: Checkout code + uses: actions/checkout@v2 + + - name: Set up Python + uses: actions/setup-python@v2 + with: + python-version: '3.x' + + - name: Install backend dependencies + run: | + python -m pip install --upgrade pip + pip install -r backend/requirements.txt + + - name: Run tests + run: | + python backend/test_blackboard_scraper.py From 7ab6ce47042472ad89509f0c2c5eaea216aa8ede Mon Sep 17 00:00:00 2001 From: Jay Date: Sun, 7 Jan 2024 15:36:05 -0500 Subject: [PATCH 03/12] Update requirements.txt --- backend/requirements.txt | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/backend/requirements.txt b/backend/requirements.txt index 3c16ea3..d5177b0 100644 --- a/backend/requirements.txt +++ b/backend/requirements.txt @@ -1,10 +1,9 @@ ghostscript selenium flask -ray argparse bs4 gunicorn flask_cors flask_apscheduler -pydrive2 \ No newline at end of file +pydrive2 From 9c373766b4e1d68f377ec9213189630fdde68b6f Mon Sep 17 00:00:00 2001 From: Jay Date: Sun, 7 Jan 2024 15:37:48 -0500 Subject: [PATCH 04/12] Update test_blackboard_scraper.py --- backend/test_blackboard_scraper.py | 333 +++++++++++++++++++++++++++-- 1 file changed, 321 insertions(+), 12 deletions(-) diff --git a/backend/test_blackboard_scraper.py b/backend/test_blackboard_scraper.py index 87479ad..b86ab67 100644 --- a/backend/test_blackboard_scraper.py +++ b/backend/test_blackboard_scraper.py @@ -1,12 +1,15 @@ -import logging import random +import time import unittest -from backend.blackboard_scraper import BlackboardSession -from unittest.mock import MagicMock, patch +from blackboard_scraper import BlackboardSession +from unittest.mock import patch from usernames import usernames + class TestBlackboardSession(unittest.TestCase): + # * Login Tests *# + def test_valid_credentials_login(self): # Set up username = 'Free8864' @@ -14,14 +17,15 @@ def test_valid_credentials_login(self): session = BlackboardSession(username=username, password=password) # Execute login - response = session.login() + session.login() + + response = session.get_response() # Check the response expected_message = "Login successful." self.assertEqual(response, expected_message) - def test_invalid_both_login(self): # Set up username = 'InvalidUsername' @@ -29,24 +33,28 @@ def test_invalid_both_login(self): session = BlackboardSession(username=username, password=password) # Execute login - response = session.login() + session.login() + + response = session.get_response() # Check the response expected_error_message = "The username you entered cannot be identified." self.assertEqual(response, expected_error_message) - def test_failed_login_invalid_password(self): # selected a random username from usernames.py username = random.choice(list(usernames)) invalid_password = 'InvalidPassword' - session = BlackboardSession(username=username, password=invalid_password) + session = BlackboardSession( + username=username, password=invalid_password) # Execute login - response = session.login() + session.login() + + response = session.get_response() # Check the response error_messages = [ @@ -55,21 +63,322 @@ def test_failed_login_invalid_password(self): self.assertTrue(response in error_messages) - def test_failed_login_invalid_username(self): # Set up invalid_username = 'InvalidUsername' password = 'InvalidPassword' - session = BlackboardSession(username=invalid_username, password=password) + session = BlackboardSession( + username=invalid_username, password=password) # Execute login - response = session.login() + session.login() + + response = session.get_response() # Check the response expected_error_message = "The username you entered cannot be identified." self.assertEqual(response, expected_error_message) + # * Enable Instructors *# + + def test_enable_instructors_logged_in(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = True + + # Mock the GET request + with patch.object(session, '_get_request') as mock_get_request: + mock_get_request.return_value.status_code = 200 + mock_get_request.return_value.content = ''' + +
+ +
+ + ''' + + # Mock the POST request + with patch.object(session, '_send_post_request') as mock_post_request: + mock_post_request.return_value.status_code = 302 + mock_post_request.return_value.headers = { + 'Location': 'https://kettering.blackboard.com'} + + # Execute enable_instructors + session.enable_instructors() + + # Check the response + self.assertTrue(session.instructorsFound) + self.assertAlmostEqual( + session.last_activity_time, time.time(), delta=1) + + def test_enable_instructors_not_logged_in(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = False + + # Execute enable_instructors + session.enable_instructors() + + # Check the response + self.assertEqual(session.response, "Not logged in.") + self.assertFalse(session.instructorsFound) + self.assertIsNone(session.last_activity_time) + + def test_enable_instructors_get_request_failed(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = True + + # Mock the GET request + with patch.object(session, '_get_request') as mock_get_request: + mock_get_request.return_value.status_code = 500 + + # Mock the logging.error function + with patch('logging.error') as mock_logging_error: + # Execute enable_instructors + session.enable_instructors() + + # Check the response + self.assertFalse(session.instructorsFound) + + # Check the logging.error call + mock_logging_error.assert_called_once_with( + f"GET request failed with status code: {mock_get_request.return_value.status_code}") + + def test_enable_instructors_post_request_failed(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = True + + # Mock the GET request + with patch.object(session, '_get_request') as mock_get_request: + mock_get_request.return_value.status_code = 200 + mock_get_request.return_value.content = ''' + +
+ +
+ + ''' + + # Mock the POST request + with patch.object(session, '_send_post_request') as mock_post_request: + mock_post_request.return_value.status_code = 500 + + # Mock the logging.error function + with patch('logging.error') as mock_logging_error: + # Execute enable_instructors + session.enable_instructors() + + # Check the response + self.assertFalse(session.instructorsFound) + + # Check the logging.error call + mock_logging_error.assert_called_once_with( + f"POST request failed with status code: {mock_post_request.return_value.status_code}") + + # * Get Courses *# + + def test_get_courses_logged_in(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = True + + # Mock the POST request + with patch.object(session, '_send_post_request') as mock_post_request: + mock_post_request.return_value.status_code = 200 + mock_post_request.return_value.content = ''' + +
+ +
+ + ''' + + # Execute get_courses + session.get_courses() + + # Check the response + expected_courses = { + 'Course 1': 'course1_link', + 'Course 2': 'course2_link' + } + self.assertEqual(session.courses, expected_courses) + self.assertAlmostEqual( + session.last_activity_time, time.time(), delta=1) + + def test_get_courses_not_logged_in(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = False + + # Execute get_courses + session.get_courses() + + # Check the response + self.assertEqual(session.response, "Not logged in.") + self.assertEqual(session.courses, {}) + self.assertIsNone(session.last_activity_time) + + def test_get_courses_no_courses(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = True + + # Mock the POST request + with patch.object(session, '_send_post_request') as mock_post_request: + mock_post_request.return_value.status_code = 200 + mock_post_request.return_value.content = ''' + +
+
    +
  • You are not currently enrolled in any courses.
  • +
+
+ + ''' + + # Execute get_courses + session.get_courses() + + # Check the response + self.assertEqual(session.response, + "You are not currently enrolled in any courses.") + self.assertEqual(session.courses, {}) + + def test_get_courses_error_finding_course_list(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = True + + # Mock the POST request + with patch.object(session, '_send_post_request') as mock_post_request: + mock_post_request.return_value.status_code = 500 + + # Mock the logging.error function + with patch('logging.error') as mock_logging_error: + # Execute get_courses + session.get_courses() + + # Check the response + self.assertIsInstance(session.response, Exception) + self.assertEqual(str(session.response), 'POST request failed.') + self.assertEqual(session.courses, {}) + mock_logging_error.assert_called_once() + + # * Get Download Tasks *# + + +def test_get_download_tasks_logged_in(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = True + session.courses = { + 'Course 1': 'course1_link', + 'Course 2': 'course2_link' + } + + with patch.object(session, '_get_request') as mock_get_request: + mock_get_request.side_effect = [ + type('', (), {'status_code': 200, 'content': ''' + + +
+ +
+ + + '''}), + type('', (), {'status_code': 200, 'content': ''' + + +
+ +
+ + + '''}) + ] + + # Call the method + session.get_download_tasks() + + # Assert the result + expected_result = [ + ('Course 1', 'Assignment 1', 'download_link1'), + ('Course 1', 'Assignment 2', 'download_link2'), + ('Course 2', 'Assignment 1', 'download_link1'), + ('Course 2', 'Assignment 2', 'download_link2'), + ] + + self.assertEqual(session.download_tasks, expected_result) + self.assertTrue(session.downloadTasksFound) + self.assertAlmostEqual( + session.last_activity_time, time.time(), delta=1) + + def test_get_download_tasks_not_logged_in(self): + # Set up + username = 'Free8864' + password = '#CFi^F6TTwot2j' + session = BlackboardSession(username=username, password=password) + session.is_logged_in = False + + # Execute get_download_tasks + session.get_download_tasks() + + # Check the response + self.assertEqual(session.response, "Not logged in.") + self.assertFalse(session.downloadTasksFound) + self.assertIsNone(session.last_activity_time) + if __name__ == '__main__': unittest.main() From dd0a15d6584b780de2c6331f5b4579649c53d7ab Mon Sep 17 00:00:00 2001 From: Jay Date: Sun, 7 Jan 2024 15:39:34 -0500 Subject: [PATCH 05/12] Update blackboard_scraper.py --- backend/blackboard_scraper.py | 146 ++++++++++++++++++---------------- 1 file changed, 78 insertions(+), 68 deletions(-) diff --git a/backend/blackboard_scraper.py b/backend/blackboard_scraper.py index 0817d5e..81d4357 100644 --- a/backend/blackboard_scraper.py +++ b/backend/blackboard_scraper.py @@ -84,6 +84,12 @@ def set_response(self, response): def get_response(self): return self.response + + def get_InstructorsFound(self): + return self.instructorsFound + + def set_InstructorsFound(self, instructorsFound): + self.instructorsFound = instructorsFound def shutdown(self): """ @@ -159,7 +165,7 @@ def scrape(self): return self.enable_instructors() - if self.instructorsFound == False: + if self.get_InstructorsFound() == False: self.response = "No instructors found." self.get_courses() @@ -189,70 +195,76 @@ def enable_instructors(self): try: get_url = "https://kettering.blackboard.com/webapps/portal/execute/tabs/tabAction?tab_tab_group_id=_1_1&forwardUrl=edit_module%2F_4_1%2Fbbcourseorg%3Fcmd%3Dedit&recallUrl=%2Fwebapps%2Fportal%2Fexecute%2Ftabs%2FtabAction%3Ftab_tab_group_id%3D_1_1" - get_response = self._get_request(get_url) - - if get_response.status_code != 200: - raise Exception("GET request failed.") - - # Using beautiful soup get the value from this input #moduleEditForm > input[type=hidden]:nth-child(1) - soup = BeautifulSoup(get_response.content, "html.parser") - nonce_value = soup.select_one( - '#moduleEditForm > input[type=hidden]:nth-child(1)')['value'] - - url = "https://kettering.blackboard.com/webapps/portal/execute/tabs/tabAction?tab_tab_group_id=_1_1&forwardUrl=proc_edit/_4_1/bbcourseorg&recallUrl=%2Fwebapps%2Fportal%2Fexecute%2Ftabs%2FtabAction%3Ftab_tab_group_id%3D_1_1" - payload = { - 'tab_tab_group_id': '_1_1', - 'forwardUrl': 'proc_edit/_4_1/bbcourseorg', - 'blackboard.platform.security.NonceUtil.nonce': nonce_value, - 'recallUrl': '/webapps/portal/execute/tabs/tabAction?tab_tab_group_id=_1_1', - 'cmd': 'processEdit', - 'serviceLevel': '', - 'termDisplayOrder': '_254_1', - 'amc.groupbyterm': 'true', - 'selectAll_254_1': 'true', - 'amc.showterm._254_1': 'true', - 'termCourses__254_1': 'true', - 'amc.showcourse._51671_1': 'true', - 'amc.showcourseid._51671_1': 'true', - 'amc.showinstructors._51671_1': 'true', - 'amc.showcourse._51672_1': 'true', - 'amc.showcourseid._51672_1': 'true', - 'amc.showinstructors._51672_1': 'true', - 'amc.showcourse._51629_1': 'true', - 'amc.showcourseid._51629_1': 'true', - 'amc.showinstructors._51629_1': 'true', - 'amc.showcourse._51904_1': 'true', - 'amc.showcourseid._51904_1': 'true', - 'amc.showinstructors._51904_1': 'true', - 'amc.showcourse._51945_1': 'true', - 'amc.showcourseid._51945_1': 'true', - 'amc.showinstructors._51945_1': 'true', - 'amc.url.name.1': '', - 'amc.url.url.1': '', - 'amc.url.name.2': '', - 'amc.url.url.2': '', - 'amc.url.name.3': '', - 'amc.url.url.3': '', - 'amc.url.name.4': '', - 'amc.url.url.4': '', - 'amc.url.name.5': '', - 'amc.url.url.5': '', - 'bottom_Submit': 'Submit' - } - enable_instructors_response = self._send_post_request( - url, data=payload, allow_redirects=False) - - if enable_instructors_response.status_code == 302: - redirected_url = enable_instructors_response.headers['Location'] - logging.info( - f"Successful POST request. Redirected to: {redirected_url}") - self.instructorsFound = True - else: - self.instructorsFound = False + try: + get_response = self._get_request(get_url) + + if get_response.status_code != 200: + raise Exception("GET request failed.") + + # Using beautiful soup get the value from this input #moduleEditForm > input[type=hidden]:nth-child(1) + soup = BeautifulSoup(get_response.content, "html.parser") + nonce_value = soup.select_one( + '#moduleEditForm > input[type=hidden]:nth-child(1)')['value'] + + url = "https://kettering.blackboard.com/webapps/portal/execute/tabs/tabAction?tab_tab_group_id=_1_1&forwardUrl=proc_edit/_4_1/bbcourseorg&recallUrl=%2Fwebapps%2Fportal%2Fexecute%2Ftabs%2FtabAction%3Ftab_tab_group_id%3D_1_1" + payload = { + 'tab_tab_group_id': '_1_1', + 'forwardUrl': 'proc_edit/_4_1/bbcourseorg', + 'blackboard.platform.security.NonceUtil.nonce': nonce_value, + 'recallUrl': '/webapps/portal/execute/tabs/tabAction?tab_tab_group_id=_1_1', + 'cmd': 'processEdit', + 'serviceLevel': '', + 'termDisplayOrder': '_254_1', + 'amc.groupbyterm': 'true', + 'selectAll_254_1': 'true', + 'amc.showterm._254_1': 'true', + 'termCourses__254_1': 'true', + 'amc.showcourse._51671_1': 'true', + 'amc.showcourseid._51671_1': 'true', + 'amc.showinstructors._51671_1': 'true', + 'amc.showcourse._51672_1': 'true', + 'amc.showcourseid._51672_1': 'true', + 'amc.showinstructors._51672_1': 'true', + 'amc.showcourse._51629_1': 'true', + 'amc.showcourseid._51629_1': 'true', + 'amc.showinstructors._51629_1': 'true', + 'amc.showcourse._51904_1': 'true', + 'amc.showcourseid._51904_1': 'true', + 'amc.showinstructors._51904_1': 'true', + 'amc.showcourse._51945_1': 'true', + 'amc.showcourseid._51945_1': 'true', + 'amc.showinstructors._51945_1': 'true', + 'amc.url.name.1': '', + 'amc.url.url.1': '', + 'amc.url.name.2': '', + 'amc.url.url.2': '', + 'amc.url.name.3': '', + 'amc.url.url.3': '', + 'amc.url.name.4': '', + 'amc.url.url.4': '', + 'amc.url.name.5': '', + 'amc.url.url.5': '', + 'bottom_Submit': 'Submit' + } + enable_instructors_response = self._send_post_request( + url, data=payload, allow_redirects=False) + + if enable_instructors_response.status_code == 302: + redirected_url = enable_instructors_response.headers['Location'] + logging.info( + f"Successful POST request. Redirected to: {redirected_url}") + self.set_InstructorsFound(True) + else: + self.set_InstructorsFound(False) + logging.error( + f"POST request failed with status code: {enable_instructors_response.status_code}") + + self.last_activity_time = time.time() + + except Exception as e: logging.error( - f"POST request failed with status code: {enable_instructors_response.status_code}") - - self.last_activity_time = time.time() + f"GET request failed with status code: {get_response.status_code}") + return except Exception as e: logging.error(f"An error occurred enabling instructors: {e}") @@ -304,7 +316,7 @@ def get_courses(self): hrefs = {course.text.strip(): course.find("a")["href"].strip() for course in courses_list if course.find("a") and course.find("a").get("href")} - if self.instructorFound: + if self.get_InstructorsFound() == True: # Process instructors and format course names for course in courses_list: try: @@ -337,6 +349,7 @@ def get_courses(self): except Exception as e: self.courseFound = False + self.response = e logging.error(f"An error occurred while getting courses: {e}") def download_and_save_file(self): @@ -470,6 +483,3 @@ def process_course(course, href): self.download_tasks = download_tasks self.downloadTasksFound = True self.last_activity_time = time.time() - - - From 7056a095b039d0c9f81017cc92b08de50161f5a0 Mon Sep 17 00:00:00 2001 From: Jay Date: Sun, 7 Jan 2024 17:39:21 -0500 Subject: [PATCH 06/12] Update docker-image.yml --- .github/workflows/docker-image.yml | 20 ++++++++++++-------- 1 file changed, 12 insertions(+), 8 deletions(-) diff --git a/.github/workflows/docker-image.yml b/.github/workflows/docker-image.yml index c1b60ed..fea233a 100644 --- a/.github/workflows/docker-image.yml +++ b/.github/workflows/docker-image.yml @@ -5,25 +5,29 @@ on: branches: [ "main" ] pull_request: branches: [ "main" ] + release: + types: [published] jobs: backend-build: - runs-on: ubuntu-latest - steps: - name: Checkout backend code uses: actions/checkout@v3 - - name: Build the Docker image for backend - run: docker build ./backend -t backend-image-name:$(date +%s) + - name: Build and push the Docker image for backend + run: | + docker build ./backend -t my-docker-hub-namespace/backend-image-name:$(date +%s) + echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin + docker push my-docker-hub-namespace/backend-image-name:$(date +%s) frontend-build: - runs-on: ubuntu-latest - steps: - name: Checkout frontend code uses: actions/checkout@v3 - - name: Build the Docker image for frontend - run: docker build ./frontend -t frontend-image-name:$(date +%s) + - name: Build and push the Docker image for frontend + run: | + docker build ./frontend -t my-docker-hub-namespace/frontend-image-name:$(date +%s) + echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin + docker push my-docker-hub-namespace/frontend-image-name:$(date +%s) From b46476f15b349059d2cc23acbd748a1509f6efa2 Mon Sep 17 00:00:00 2001 From: Jaydin_MacBook Date: Sun, 7 Jan 2024 17:46:57 -0500 Subject: [PATCH 07/12] Update docker-image.yml --- .github/workflows/docker-image.yml | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/.github/workflows/docker-image.yml b/.github/workflows/docker-image.yml index fea233a..f4c6a9c 100644 --- a/.github/workflows/docker-image.yml +++ b/.github/workflows/docker-image.yml @@ -15,19 +15,23 @@ jobs: steps: - name: Checkout backend code uses: actions/checkout@v3 + with: + path: 'backend' - name: Build and push the Docker image for backend run: | - docker build ./backend -t my-docker-hub-namespace/backend-image-name:$(date +%s) + docker build ${{ github.workspace }}/backend -t themanwholikestocode/archive-me-prod:backend-$(date +%s) echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin - docker push my-docker-hub-namespace/backend-image-name:$(date +%s) + docker push themanwholikestocode/archive-me-prod:backend-$(date +%s) frontend-build: runs-on: ubuntu-latest steps: - name: Checkout frontend code uses: actions/checkout@v3 + with: + path: 'frontend' - name: Build and push the Docker image for frontend run: | - docker build ./frontend -t my-docker-hub-namespace/frontend-image-name:$(date +%s) + docker build ${{ github.workspace }}/frontend -t themanwholikestocode/archive-me-prod:frontend-$(date +%s) echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin - docker push my-docker-hub-namespace/frontend-image-name:$(date +%s) + docker push themanwholikestocode/archive-me-prod:frontend-$(date +%s) From 3006eb57d8ac6ea117e20476a219dbe1c6a0604c Mon Sep 17 00:00:00 2001 From: Jaydin_MacBook Date: Sun, 7 Jan 2024 17:49:11 -0500 Subject: [PATCH 08/12] Update docker-image.yml --- .github/workflows/docker-image.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/docker-image.yml b/.github/workflows/docker-image.yml index f4c6a9c..e0b1019 100644 --- a/.github/workflows/docker-image.yml +++ b/.github/workflows/docker-image.yml @@ -19,7 +19,7 @@ jobs: path: 'backend' - name: Build and push the Docker image for backend run: | - docker build ${{ github.workspace }}/backend -t themanwholikestocode/archive-me-prod:backend-$(date +%s) + docker build /backend -t themanwholikestocode/archive-me-prod:backend-$(date +%s) echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin docker push themanwholikestocode/archive-me-prod:backend-$(date +%s) @@ -32,6 +32,6 @@ jobs: path: 'frontend' - name: Build and push the Docker image for frontend run: | - docker build ${{ github.workspace }}/frontend -t themanwholikestocode/archive-me-prod:frontend-$(date +%s) + docker build /frontend -t themanwholikestocode/archive-me-prod:frontend-$(date +%s) echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin docker push themanwholikestocode/archive-me-prod:frontend-$(date +%s) From fabb63af5d366f74db38a6e9693e056db615e8d6 Mon Sep 17 00:00:00 2001 From: Jaydin_MacBook Date: Sun, 7 Jan 2024 17:50:34 -0500 Subject: [PATCH 09/12] Update docker-image.yml --- .github/workflows/docker-image.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/docker-image.yml b/.github/workflows/docker-image.yml index e0b1019..3fdb0bb 100644 --- a/.github/workflows/docker-image.yml +++ b/.github/workflows/docker-image.yml @@ -19,7 +19,7 @@ jobs: path: 'backend' - name: Build and push the Docker image for backend run: | - docker build /backend -t themanwholikestocode/archive-me-prod:backend-$(date +%s) + docker build ./backend -t themanwholikestocode/archive-me-prod:backend-$(date +%s) echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin docker push themanwholikestocode/archive-me-prod:backend-$(date +%s) @@ -32,6 +32,6 @@ jobs: path: 'frontend' - name: Build and push the Docker image for frontend run: | - docker build /frontend -t themanwholikestocode/archive-me-prod:frontend-$(date +%s) + docker build ./frontend -t themanwholikestocode/archive-me-prod:frontend-$(date +%s) echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin docker push themanwholikestocode/archive-me-prod:frontend-$(date +%s) From 0f9551da92ca360e0cdd343f370ff445fac48a65 Mon Sep 17 00:00:00 2001 From: Jay Date: Sun, 7 Jan 2024 19:11:04 -0500 Subject: [PATCH 10/12] Update docker-image.yml --- .github/workflows/docker-image.yml | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/.github/workflows/docker-image.yml b/.github/workflows/docker-image.yml index 3fdb0bb..52600c7 100644 --- a/.github/workflows/docker-image.yml +++ b/.github/workflows/docker-image.yml @@ -15,8 +15,7 @@ jobs: steps: - name: Checkout backend code uses: actions/checkout@v3 - with: - path: 'backend' + - name: Build and push the Docker image for backend run: | docker build ./backend -t themanwholikestocode/archive-me-prod:backend-$(date +%s) @@ -28,8 +27,7 @@ jobs: steps: - name: Checkout frontend code uses: actions/checkout@v3 - with: - path: 'frontend' + - name: Build and push the Docker image for frontend run: | docker build ./frontend -t themanwholikestocode/archive-me-prod:frontend-$(date +%s) From 11eb1ac777d436bdcbe0826133863e65e6f123d3 Mon Sep 17 00:00:00 2001 From: Jay Date: Sun, 7 Jan 2024 19:31:09 -0500 Subject: [PATCH 11/12] Update docker-image.yml --- .github/workflows/docker-image.yml | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/.github/workflows/docker-image.yml b/.github/workflows/docker-image.yml index 52600c7..3672109 100644 --- a/.github/workflows/docker-image.yml +++ b/.github/workflows/docker-image.yml @@ -18,9 +18,12 @@ jobs: - name: Build and push the Docker image for backend run: | - docker build ./backend -t themanwholikestocode/archive-me-prod:backend-$(date +%s) + TIMESTAMP=$(date +%s) + docker build ./backend -t themanwholikestocode/archive-me-prod:backend-$TIMESTAMP + docker tag themanwholikestocode/archive-me-prod:backend-$TIMESTAMP domain.com/repo/tag_docker_name:latest echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin - docker push themanwholikestocode/archive-me-prod:backend-$(date +%s) + docker push themanwholikestocode/archive-me-prod:backend-$TIMESTAMP + docker push domain.com/repo/tag_docker_name:latest frontend-build: runs-on: ubuntu-latest @@ -30,6 +33,9 @@ jobs: - name: Build and push the Docker image for frontend run: | - docker build ./frontend -t themanwholikestocode/archive-me-prod:frontend-$(date +%s) + TIMESTAMP=$(date +%s) + docker build ./frontend -t themanwholikestocode/archive-me-prod:frontend-$TIMESTAMP + docker tag themanwholikestocode/archive-me-prod:frontend-$TIMESTAMP domain.com/repo/tag_docker_name:latest echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin - docker push themanwholikestocode/archive-me-prod:frontend-$(date +%s) + docker push themanwholikestocode/archive-me-prod:frontend-$TIMESTAMP + docker push domain.com/repo/tag_docker_name:latest From 7c84416d7673fd879cc9bc40ccabd8dff3bcafd3 Mon Sep 17 00:00:00 2001 From: Jay Date: Sun, 7 Jan 2024 19:37:16 -0500 Subject: [PATCH 12/12] Update docker-image.yml --- .github/workflows/docker-image.yml | 2 -- 1 file changed, 2 deletions(-) diff --git a/.github/workflows/docker-image.yml b/.github/workflows/docker-image.yml index 3672109..cb56431 100644 --- a/.github/workflows/docker-image.yml +++ b/.github/workflows/docker-image.yml @@ -23,7 +23,6 @@ jobs: docker tag themanwholikestocode/archive-me-prod:backend-$TIMESTAMP domain.com/repo/tag_docker_name:latest echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin docker push themanwholikestocode/archive-me-prod:backend-$TIMESTAMP - docker push domain.com/repo/tag_docker_name:latest frontend-build: runs-on: ubuntu-latest @@ -38,4 +37,3 @@ jobs: docker tag themanwholikestocode/archive-me-prod:frontend-$TIMESTAMP domain.com/repo/tag_docker_name:latest echo ${{ secrets.DOCKER_PASSWORD }} | docker login -u ${{ secrets.DOCKER_USERNAME }} --password-stdin docker push themanwholikestocode/archive-me-prod:frontend-$TIMESTAMP - docker push domain.com/repo/tag_docker_name:latest