From 6aed0e60c0fe4beda44a0d104a2b81da92e317eb Mon Sep 17 00:00:00 2001 From: Pankaj K Borade <62588059+BkPankaj@users.noreply.github.com> Date: Sat, 8 Jun 2024 15:31:18 +0530 Subject: [PATCH] Combined PR of ROS1 laser block and CI Workflow with basic frontend test (#309) * added ROS1 Laser block * added npm install yml code * changed node version * adding legacy in npm install * added backend installation * added CI workflow with basic frontend test * added permission for node modules * changed node version to 18 * modified node modules with cache * added opening google.com using selenium in standalone-chrome ~ first test * added automatic Opening VC and clicking File and Open button of menubar * Update collection-factory to include blocks in the UI --------- Co-authored-by: toshan-luktuke --- .github/first_test.py | 55 +++ .github/workflows/CI_Workflow.yml | 93 ++++ frontend/src/App.test.tsx | 4 +- .../blocks/collection/collection-factory.tsx | 6 +- .../collection/ros-sensors/ROSLaserScan.json | 396 ++++++++++++++++++ 5 files changed, 551 insertions(+), 3 deletions(-) create mode 100644 .github/first_test.py create mode 100644 .github/workflows/CI_Workflow.yml create mode 100644 frontend/src/components/blocks/collection/ros-sensors/ROSLaserScan.json diff --git a/.github/first_test.py b/.github/first_test.py new file mode 100644 index 00000000..e473bc00 --- /dev/null +++ b/.github/first_test.py @@ -0,0 +1,55 @@ +from selenium import webdriver +from selenium.webdriver.common.by import By +from selenium.webdriver.support.ui import WebDriverWait +from selenium.webdriver.support import expected_conditions as EC +import time + +try: + # Set up the webdriver to connect to the remote Selenium server + options = webdriver.ChromeOptions() + options.add_argument('--no-sandbox') + options.add_argument('--disable-dev-shm-usage') + + # Disable headless mode to show the Chrome UI + # options.add_argument('--headless=false') + + # Remote WebDriver URL (provided by the selenium/standalone-chrome service) + driver = webdriver.Remote( + command_executor='http://localhost:4444/wd/hub', + options=options + ) + + # Open the browser and go to the URL + driver.get('http://IP:4000') + + # time.sleep(120) + + + # Wait for the "File" button to be clickable and click it + basic_button = WebDriverWait(driver, 20).until( + EC.element_to_be_clickable((By.XPATH, "//button[contains(@class, 'menu-button') and .//span[text()='File']]")) + ) + basic_button.click() + + + # Wait for the dropdown menu to be visible + dropdown_menu = WebDriverWait(driver, 10).until( + EC.visibility_of_element_located((By.XPATH, "//ul[@role='menu' and @aria-label='File']")) + ) + + # Wait for the "Open" menu item to be clickable and click it + code_menu_item = WebDriverWait(driver, 20).until( + EC.element_to_be_clickable((By.XPATH, "//ul[@role='menu' and @aria-label='File']//li[text()='Open']")) + ) + code_menu_item.click() + + time.sleep(20) + + + # Capture a screenshot + driver.get_screenshot_as_file('screenshot.png') + +finally: + # Close the browser + if driver: + driver.quit() diff --git a/.github/workflows/CI_Workflow.yml b/.github/workflows/CI_Workflow.yml new file mode 100644 index 00000000..3ff45e6a --- /dev/null +++ b/.github/workflows/CI_Workflow.yml @@ -0,0 +1,93 @@ +name: CI Workflow + +on: + pull_request: + branches: + - 'master' + +jobs: + setup: + runs-on: ubuntu-latest + + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Set up Node.js + uses: actions/setup-node@v4 + with: + node-version: '18' + + - name: Install frontend dependencies + if: steps.cache-node-modules.outputs.cache-hit != 'true' + working-directory: frontend + run: npm ci --legacy-peer-deps + + - name: Cache Node modules + if: steps.cache-node-modules.outputs.cache-hit != 'true' + uses: actions/cache@v4 + with: + path: frontend/node_modules + key: ${{ runner.os }}-node-${{ hashFiles('**/frontend/package-lock.json') }} + restore-keys: | + ${{ runner.os }}-node- + + + - name: Set up Python + uses: actions/setup-python@v5 + with: + python-version: '3.x' + + - name: Create virtual environment + run: python -m venv .venv + + - name: Activate virtual environment + run: source .venv/bin/activate + + - name: Install dependencies + run: | + python -m pip install --upgrade pip + pip install -r backend/requirements.txt + + - name: Add .env file + run: cp backend/.env.template backend/.env + + - name: Generate static files + run: python backend/manage.py collectstatic + + - name: Save venv + uses: actions/upload-artifact@v4 + with: + name: venv + path: .venv + + frontend-tests: + runs-on: ubuntu-latest + needs: setup + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Restore Node modules cache + id: restore-node-modules + uses: actions/cache@v4 + with: + path: frontend/node_modules + key: ${{ runner.os }}-node-${{ hashFiles('**/frontend/package-lock.json') }} + + - name: Install frontend dependencies + if: steps.restore-node-modules.outputs.cache-hit != 'true' + working-directory: frontend + run: npm ci --legacy-peer-deps + + - name: Permissions for node_modules + run: chmod -R +x frontend/node_modules/.bin + + - name: Verify node_modules restoration + run: | + ls -la frontend/node_modules + ls -la frontend/node_modules/.bin + + - name: Run frontend test + run: npm test -- --watchAll=false + working-directory: frontend \ No newline at end of file diff --git a/frontend/src/App.test.tsx b/frontend/src/App.test.tsx index 2a68616d..e9d70bf6 100644 --- a/frontend/src/App.test.tsx +++ b/frontend/src/App.test.tsx @@ -2,8 +2,8 @@ import React from 'react'; import { render, screen } from '@testing-library/react'; import App from './App'; -test('renders learn react link', () => { +test('renders App for test', () => { render(); - const linkElement = screen.getByText(/learn react/i); + const linkElement = screen.getByText(/File/i); expect(linkElement).toBeInTheDocument(); }); diff --git a/frontend/src/components/blocks/collection/collection-factory.tsx b/frontend/src/components/blocks/collection/collection-factory.tsx index 45104f6e..19b69676 100644 --- a/frontend/src/components/blocks/collection/collection-factory.tsx +++ b/frontend/src/components/blocks/collection/collection-factory.tsx @@ -109,7 +109,8 @@ export const collectionBlocks: { 'blocks': CollectionBlockType, 'children': { 'cameraRos': {'label': 'CameraROS'}, 'odometer': {'label': 'Odometer'}, - 'imu': {'label': 'IMU'} + 'imu': {'label': 'IMU'}, + 'laserRos':{'label':'LaserROS'} } }, 'ros2sensors': { @@ -170,12 +171,15 @@ export function getCollectionBlock(name: string) { return import('./ros-sensors/Odometer.json'); case 'drivers.rossensors.imu': return import('./ros-sensors/IMU.json'); + case 'drivers.rossensors.laserRos': + return import('./ros-sensors/ROSLaserScan.json'); case 'drivers.ros2sensors.cameraRos2': return import('./ros-sensors/ROS2Camera.json'); case 'drivers.ros2sensors.laserRos2': return import('./ros-sensors/ROS2LaserScan.json'); case 'processing.tensorflow.objectDetector': return import('./tensorflow/ObjectDetector.json'); + default: break; diff --git a/frontend/src/components/blocks/collection/ros-sensors/ROSLaserScan.json b/frontend/src/components/blocks/collection/ros-sensors/ROSLaserScan.json new file mode 100644 index 00000000..552fcf60 --- /dev/null +++ b/frontend/src/components/blocks/collection/ros-sensors/ROSLaserScan.json @@ -0,0 +1,396 @@ +{ + "editor": { + "id": "6168472f-f6d8-45cf-bc98-3051074065d5", + "locked": false, + "offsetX": -254.49013919915978, + "offsetY": -42.460178287148096, + "zoom": 86.87812499999998, + "gridSize": 0, + "layers": [ + { + "id": "4ed5b2c0-ea15-4800-aa27-5a6509869898", + "type": "diagram-links", + "isSvg": true, + "transformed": true, + "models": { + "89196a6e-d40b-4b8f-b499-96a43299657c": { + "id": "89196a6e-d40b-4b8f-b499-96a43299657c", + "type": "default", + "selected": false, + "source": "0005-248b709f-e6c6-4b8d-94f1-d986193b7cad", + "sourcePort": "8563511b-b3d0-4833-9426-a929f9d72a85", + "target": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "targetPort": "7c302710-552a-4be3-aa87-47143ad3442e", + "points": [ + { + "id": "9cab4f63-8c44-4be8-ba68-8cdaefe623a2", + "type": "point", + "x": 983.8381516626423, + "y": 137.42643805645446 + }, + { + "id": "d47a6102-e816-4e13-90ef-c8b655d8e62f", + "type": "point", + "x": 1317.2937968058027, + "y": 363.1449356346666 + } + ], + "labels": [], + "width": 3, + "color": "gray", + "curvyness": 50, + "selectedColor": "rgb(0,192,255)" + }, + "2cfa2c28-af53-4daf-a364-162e74775831": { + "id": "2cfa2c28-af53-4daf-a364-162e74775831", + "type": "default", + "selected": false, + "source": "c9fda9bb-631f-45dd-9551-47f856966126", + "sourcePort": "03136f30-35dd-4713-bac2-afc19626ed88", + "target": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "targetPort": "258fd7ac-4b13-468f-b5bf-5bfb84e6b1ba", + "points": [ + { + "id": "4855eb9e-572a-47d6-a99f-0a56107014b0", + "type": "point", + "x": 677.1633130776993, + "y": 497.3750864161133 + }, + { + "id": "47449955-47be-407c-850d-a9ac04bcd565", + "type": "point", + "x": 1018.8105124252947, + "y": 609.5437274946717 + } + ], + "labels": [], + "width": 3, + "color": "gray", + "curvyness": 50, + "selectedColor": "rgb(0,192,255)" + }, + "556ded65-b41d-41e5-a4ab-7b97400ab48c": { + "id": "556ded65-b41d-41e5-a4ab-7b97400ab48c", + "type": "default", + "selected": false, + "source": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "sourcePort": "64c11722-3520-4eee-a740-b3311361d269", + "target": "d69e6f41-3d18-4e5d-b747-69dd613dd6ef", + "targetPort": "75040235-29aa-460e-9cb4-623ccd47a144", + "points": [ + { + "id": "d6b79658-7bae-44af-9555-a630f2860f9c", + "type": "point", + "x": 1615.7962341197708, + "y": 609.5437274946717 + }, + { + "id": "4d31b78b-7efb-4ddb-ba14-9bfc89d96476", + "type": "point", + "x": 1777.1141916432186, + "y": 520.1848690003728 + } + ], + "labels": [], + "width": 3, + "color": "gray", + "curvyness": 50, + "selectedColor": "rgb(0,192,255)" + } + } + }, + { + "id": "d392083a-f18f-44c0-92da-fd7de9ccfef9", + "type": "diagram-nodes", + "isSvg": false, + "transformed": true, + "models": { + "0d952efb-e25e-41dc-8ee1-6ca00db1a831": { + "id": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "type": "basic.code", + "selected": false, + "x": 1010.4923703139029, + "y": 285.95298846293565, + "ports": [ + { + "id": "258fd7ac-4b13-468f-b5bf-5bfb84e6b1ba", + "type": "port.input", + "x": 1011.3095878857098, + "y": 602.0427941733658, + "name": "Enable", + "alignment": "left", + "parentNode": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "links": [ + "2cfa2c28-af53-4daf-a364-162e74775831" + ], + "in": true, + "label": "Enable", + "hideLabel": false + }, + { + "id": "64c11722-3520-4eee-a740-b3311361d269", + "type": "port.output", + "x": 1608.295300798465, + "y": 602.0427941733658, + "name": "Out", + "alignment": "right", + "parentNode": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "links": [ + "556ded65-b41d-41e5-a4ab-7b97400ab48c" + ], + "in": false, + "label": "Out", + "hideLabel": false + }, + { + "id": "7c302710-552a-4be3-aa87-47143ad3442e", + "type": "port.parameter", + "x": 1309.7928722662177, + "y": 355.6440023133607, + "name": "ROSTopic", + "alignment": "top", + "parentNode": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "links": [ + "89196a6e-d40b-4b8f-b499-96a43299657c" + ], + "in": true, + "label": "ROSTopic", + "hideLabel": false + } + ], + "data": { + "code": "#!/usr/bin/env python\n\nimport rospy\nfrom sensor_msgs.msg import LaserScan\n\nmeasure = None\n\ndef callback(msg):\n global measure\n\n measure = []\n for i in range(len(msg.ranges)):\n measure.extend((str(msg.ranges[i]),))\n\ndef main(inputs, outputs, parameters, synchronise):\n global measure\n\n try:\n rospy.init_node('laser_subscriber', anonymous=True)\n topic = parameters.read_string(\"ROSTopic\")\n rospy.Subscriber(topic, LaserScan, callback)\n\n auto_enable = False\n try:\n enable = inputs.read_number(\"Enable\")\n except Exception:\n auto_enable = True\n\n while (auto_enable or inputs.read_number('Enable') and not rospy.is_shutdown():\n if measure is not None:\n outputs.share_array(\"Out\", measure)\n measure = None \n\n synchronise()\n\n finally:\n synchronise()\n", + "frequency": "30", + "params": [ + { + "name": "ROSTopic" + } + ], + "ports": { + "in": [ + { + "name": "Enable" + } + ], + "out": [ + { + "name": "Out" + } + ] + }, + "size": { + "width": "809px", + "height": "300px" + } + } + }, + "0005-248b709f-e6c6-4b8d-94f1-d986193b7cad": { + "id": "0005-248b709f-e6c6-4b8d-94f1-d986193b7cad", + "type": "basic.constant", + "selected": false, + "x": 923.0280066509573, + "y": 21.93498493324092, + "ports": [ + { + "id": "8563511b-b3d0-4833-9426-a929f9d72a85", + "type": "port.output", + "x": 976.3372271230573, + "y": 129.92551351686953, + "name": "constant-out", + "alignment": "bottom", + "parentNode": "0005-248b709f-e6c6-4b8d-94f1-d986193b7cad", + "links": [ + "89196a6e-d40b-4b8f-b499-96a43299657c" + ], + "in": false, + "label": "ROSTopic", + "hideLabel": true + } + ], + "data": { + "name": "ROSTopic", + "value": "/scan", + "local": true + } + }, + "c9fda9bb-631f-45dd-9551-47f856966126": { + "id": "c9fda9bb-631f-45dd-9551-47f856966126", + "type": "basic.input", + "selected": false, + "x": 583.8709677419363, + "y": 470.9677419354845, + "ports": [ + { + "id": "03136f30-35dd-4713-bac2-afc19626ed88", + "type": "port.output", + "x": 669.6623885381143, + "y": 489.8741530948074, + "name": "input-out", + "alignment": "right", + "parentNode": "c9fda9bb-631f-45dd-9551-47f856966126", + "links": [ + "2cfa2c28-af53-4daf-a364-162e74775831" + ], + "in": false, + "label": "Enable", + "hideLabel": true + } + ], + "data": { + "name": "Enable" + } + }, + "d69e6f41-3d18-4e5d-b747-69dd613dd6ef": { + "id": "d69e6f41-3d18-4e5d-b747-69dd613dd6ef", + "type": "basic.output", + "selected": false, + "x": 1768.8229478865371, + "y": 493.7829458443716, + "ports": [ + { + "id": "75040235-29aa-460e-9cb4-623ccd47a144", + "type": "port.input", + "x": 1769.6132671036337, + "y": 512.683935679067, + "name": "output-in", + "alignment": "left", + "parentNode": "d69e6f41-3d18-4e5d-b747-69dd613dd6ef", + "links": [ + "556ded65-b41d-41e5-a4ab-7b97400ab48c" + ], + "in": true, + "label": "output-in", + "hideLabel": true + } + ], + "data": { + "name": "Out" + } + } + } + } + ] + }, + "version": "3.0", + "package": { + "name": "LaserROS1", + "version": "1.0", + "description": "Reads laser data and shares an array with its measures", + "author": "Pankaj Borade", + "image": "" + }, + "design": { + "board": "Python3-Noetic", + "graph": { + "blocks": [ + { + "id": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "type": "basic.code", + "data": { + "code": "#!/usr/bin/env python\n\nimport rospy\nfrom sensor_msgs.msg import LaserScan\n\nmeasure = None\n\ndef callback(msg):\n global measure\n\n measure = []\n for i in range(len(msg.ranges)):\n measure.extend((str(msg.ranges[i]),))\n\ndef main(inputs, outputs, parameters, synchronise):\n global measure\n\n try:\n rospy.init_node('laser_subscriber', anonymous=True)\n topic = parameters.read_string(\"ROSTopic\")\n rospy.Subscriber(topic, LaserScan, callback)\n\n auto_enable = False\n try:\n enable = inputs.read_number(\"Enable\")\n except Exception:\n auto_enable = True\n\n while (auto_enable or inputs.read_number('Enable') and not rospy.is_shutdown()):\n if measure is not None:\n outputs.share_array(\"Out\", measure)\n measure = None \n\n synchronise()\n\n finally:\n synchronise()\n", + "frequency": "30", + "params": [ + { + "name": "ROSTopic" + } + ], + "ports": { + "in": [ + { + "name": "Enable" + } + ], + "out": [ + { + "name": "Out" + } + ] + }, + "size": { + "width": "809px", + "height": "300px" + } + }, + "position": { + "x": 1010.4923703139029, + "y": 285.95298846293565 + } + }, + { + "id": "0005-248b709f-e6c6-4b8d-94f1-d986193b7cad", + "type": "basic.constant", + "data": { + "name": "ROSTopic", + "value": "/scan", + "local": true + }, + "position": { + "x": 923.0280066509573, + "y": 21.93498493324092 + } + }, + { + "id": "c9fda9bb-631f-45dd-9551-47f856966126", + "type": "basic.input", + "data": { + "name": "Enable" + }, + "position": { + "x": 583.8709677419363, + "y": 470.9677419354845 + } + }, + { + "id": "d69e6f41-3d18-4e5d-b747-69dd613dd6ef", + "type": "basic.output", + "data": { + "name": "Out" + }, + "position": { + "x": 1768.8229478865371, + "y": 493.7829458443716 + } + } + ], + "wires": [ + { + "source": { + "block": "0005-248b709f-e6c6-4b8d-94f1-d986193b7cad", + "port": "constant-out", + "name": "ROSTopic" + }, + "target": { + "block": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "port": "ROSTopic", + "name": "ROSTopic" + } + }, + { + "source": { + "block": "c9fda9bb-631f-45dd-9551-47f856966126", + "port": "input-out", + "name": "Enable" + }, + "target": { + "block": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "port": "Enable", + "name": "Enable" + } + }, + { + "source": { + "block": "0d952efb-e25e-41dc-8ee1-6ca00db1a831", + "port": "Out", + "name": "Out" + }, + "target": { + "block": "d69e6f41-3d18-4e5d-b747-69dd613dd6ef", + "port": "output-in", + "name": "output-in" + } + } + ] + } + }, + "dependencies": {} +} \ No newline at end of file