From 29d7d79a75f4d616eeb5ecabb36bbfbf479ee8d7 Mon Sep 17 00:00:00 2001 From: Brayan Almonte Date: Mon, 17 Jun 2024 13:53:19 -0400 Subject: [PATCH 1/7] feat(hardware): added safety relay active state to HepaUVState and safety_relay_inactive error (#15311) --- .../firmware_bindings/constants.py | 2 ++ .../firmware_bindings/messages/payloads.py | 2 ++ .../hardware_control/hepa_uv_settings.py | 3 ++ .../hardware_control/test_hepauv_settings.py | 28 ++++++++++++++++--- 4 files changed, 31 insertions(+), 4 deletions(-) diff --git a/hardware/opentrons_hardware/firmware_bindings/constants.py b/hardware/opentrons_hardware/firmware_bindings/constants.py index cd91ced91b7..4dd7c759782 100644 --- a/hardware/opentrons_hardware/firmware_bindings/constants.py +++ b/hardware/opentrons_hardware/firmware_bindings/constants.py @@ -4,6 +4,7 @@ by default. Please do not unconditionally import things outside the python standard library. """ + from enum import Enum, unique from typing import Union, Dict, List @@ -294,6 +295,7 @@ class ErrorCode(int, Enum): door_open = 0x0E reed_open = 0x0F motor_driver_error_detected = 0x10 + safety_relay_inactive = 0x11 @unique diff --git a/hardware/opentrons_hardware/firmware_bindings/messages/payloads.py b/hardware/opentrons_hardware/firmware_bindings/messages/payloads.py index c351495ba5b..50890187fe8 100644 --- a/hardware/opentrons_hardware/firmware_bindings/messages/payloads.py +++ b/hardware/opentrons_hardware/firmware_bindings/messages/payloads.py @@ -1,4 +1,5 @@ """Payloads of can bus messages.""" + # TODO (amit, 2022-01-26): Figure out why using annotations import ruins # dataclass fields interpretation. # from __future__ import annotations @@ -684,6 +685,7 @@ class GetHepaUVStatePayloadResponse(EmptyPayload): uv_light_on: utils.UInt8Field remaining_time_s: utils.UInt32Field uv_current_ma: utils.UInt16Field + safety_relay_active: utils.UInt8Field @dataclass(eq=False) diff --git a/hardware/opentrons_hardware/hardware_control/hepa_uv_settings.py b/hardware/opentrons_hardware/hardware_control/hepa_uv_settings.py index 2812cdf3f7d..6e87c466b7a 100644 --- a/hardware/opentrons_hardware/hardware_control/hepa_uv_settings.py +++ b/hardware/opentrons_hardware/hardware_control/hepa_uv_settings.py @@ -1,4 +1,5 @@ """Utilities for controlling the hepa/uv extension module.""" + import logging import asyncio from typing import Optional @@ -46,6 +47,7 @@ class HepaUVState: uv_duration_s: int remaining_time_s: int uv_current_ma: int + safety_relay_active: bool async def set_hepa_fan_state( @@ -136,6 +138,7 @@ def _listener(message: MessageDefinition, arb_id: ArbitrationId) -> None: uv_duration_s=int(message.payload.uv_duration_s.value), remaining_time_s=int(message.payload.remaining_time_s.value), uv_current_ma=int(message.payload.uv_current_ma.value), + safety_relay_active=bool(message.payload.safety_relay_active.value), ) def _filter(arb_id: ArbitrationId) -> bool: diff --git a/hardware/tests/opentrons_hardware/hardware_control/test_hepauv_settings.py b/hardware/tests/opentrons_hardware/hardware_control/test_hepauv_settings.py index dcaf85a8653..ad188bd153c 100644 --- a/hardware/tests/opentrons_hardware/hardware_control/test_hepauv_settings.py +++ b/hardware/tests/opentrons_hardware/hardware_control/test_hepauv_settings.py @@ -55,6 +55,7 @@ def create_hepa_uv_state_response( duration: int, remaining_time: int, uv_current: int, + safety_relay_active: bool, ) -> MessageDefinition: """Create a GetHepaUVStateResponse.""" return md.GetHepaUVStateResponse( @@ -63,6 +64,7 @@ def create_hepa_uv_state_response( uv_duration_s=UInt32Field(duration), remaining_time_s=UInt32Field(remaining_time), uv_current_ma=UInt16Field(uv_current), + safety_relay_active=UInt8Field(safety_relay_active), ) ) @@ -162,12 +164,29 @@ def responder( [ ( NodeId.host, - create_hepa_uv_state_response(True, 900, 300, 3300), + create_hepa_uv_state_response(True, 900, 300, 3300, True), + NodeId.hepa_uv, + ), + ( + NodeId.host, + create_hepa_uv_state_response(True, 0, 0, 33000, True), + NodeId.hepa_uv, + ), + ( + NodeId.host, + create_hepa_uv_state_response(True, 0, 0, 33000, False), + NodeId.hepa_uv, + ), + ( + NodeId.host, + create_hepa_uv_state_response(False, 0, 0, 0, True), + NodeId.hepa_uv, + ), + ( + NodeId.host, + create_hepa_uv_state_response(False, 900, 0, 0, False), NodeId.hepa_uv, ), - (NodeId.host, create_hepa_uv_state_response(True, 0, 0, 33000), NodeId.hepa_uv), - (NodeId.host, create_hepa_uv_state_response(False, 0, 0, 0), NodeId.hepa_uv), - (NodeId.host, create_hepa_uv_state_response(False, 900, 0, 0), NodeId.hepa_uv), ], ) async def test_get_hepa_uv_state( @@ -202,6 +221,7 @@ def responder( int(payload.uv_duration_s.value), int(payload.remaining_time_s.value), int(payload.uv_current_ma.value), + bool(payload.safety_relay_active.value), ) == res ) From 1193c2921666a46cb6a97b54400cdba9357fc556 Mon Sep 17 00:00:00 2001 From: Ed Cormany Date: Wed, 26 Jun 2024 16:15:58 -0400 Subject: [PATCH 2/7] chore: 7.4.0 release notes (#15531) # Overview Indicate that 7.4.0 adds support for HEPA/UV. # Test Plan Check in-app after alpha 0 tag is cut. # Changelog app-shell and api notes. # Risk assessment nil --- api/release-notes.md | 8 ++++++++ app-shell/build/release-notes.md | 8 ++++++++ 2 files changed, 16 insertions(+) diff --git a/api/release-notes.md b/api/release-notes.md index 4ba3074bf31..5c433e6cba8 100644 --- a/api/release-notes.md +++ b/api/release-notes.md @@ -6,6 +6,14 @@ log][]. For a list of currently known issues, please see the [Opentrons issue tr --- +## Opentrons Robot Software Changes in 7.4.0 + +Welcome to the v7.4.0 release of the Opentrons robot software! + +This release adds support for the [Opentrons Flex HEPA/UV Module](https://opentrons.com/products/opentrons-flex-hepa-uv-module). + +--- + ## Opentrons Robot Software Changes in 7.3.1 Welcome to the v7.3.1 release of the Opentrons robot software! diff --git a/app-shell/build/release-notes.md b/app-shell/build/release-notes.md index eadc8b068b2..60adacf1a4a 100644 --- a/app-shell/build/release-notes.md +++ b/app-shell/build/release-notes.md @@ -6,6 +6,14 @@ log][]. For a list of currently known issues, please see the [Opentrons issue tr --- +## Opentrons App Changes in 7.4.0 + +Welcome to the v7.4.0 release of the Opentrons App! + +This release adds support for the [Opentrons Flex HEPA/UV Module](https://opentrons.com/products/opentrons-flex-hepa-uv-module). + +--- + ## Opentrons App Changes in 7.3.1 Welcome to the v7.3.1 release of the Opentrons App! From 9b8eafc442abffb1cce857fecb1a5cdf1e878205 Mon Sep 17 00:00:00 2001 From: Shlok Amin Date: Thu, 27 Jun 2024 13:34:54 -0400 Subject: [PATCH 3/7] fix(api): Parse all RTP fields strictly to fix flakiness (#15187) (#15536) closes https://opentrons.atlassian.net/browse/RESC-287 --------- Co-authored-by: Max Marrone Co-authored-by: Edward Cormany Co-authored-by: Sanniti Pimpley --- api/release-notes.md | 3 ++ api/src/opentrons/protocol_engine/types.py | 44 ++++++++++++++-------- 2 files changed, 31 insertions(+), 16 deletions(-) diff --git a/api/release-notes.md b/api/release-notes.md index 5c433e6cba8..dbfbfc5bad4 100644 --- a/api/release-notes.md +++ b/api/release-notes.md @@ -12,6 +12,9 @@ Welcome to the v7.4.0 release of the Opentrons robot software! This release adds support for the [Opentrons Flex HEPA/UV Module](https://opentrons.com/products/opentrons-flex-hepa-uv-module). +### Bug Fixes + +- Fixed certain string runtime parameter values being misinterpreted as an incorrect type. --- ## Opentrons Robot Software Changes in 7.3.1 diff --git a/api/src/opentrons/protocol_engine/types.py b/api/src/opentrons/protocol_engine/types.py index 13e9515e447..77ab6231b71 100644 --- a/api/src/opentrons/protocol_engine/types.py +++ b/api/src/opentrons/protocol_engine/types.py @@ -4,7 +4,15 @@ from datetime import datetime from enum import Enum from dataclasses import dataclass -from pydantic import BaseModel, Field, validator +from pydantic import ( + BaseModel, + Field, + StrictBool, + StrictFloat, + StrictInt, + StrictStr, + validator, +) from typing import Optional, Union, List, Dict, Any, NamedTuple, Tuple, FrozenSet from typing_extensions import Literal, TypeGuard @@ -877,12 +885,14 @@ def from_hw_state(cls, state: HwTipStateType) -> "TipPresenceStatus": class RTPBase(BaseModel): """Parameters defined in a protocol.""" - displayName: str = Field(..., description="Display string for the parameter.") - variableName: str = Field(..., description="Python variable name of the parameter.") - description: Optional[str] = Field( + displayName: StrictStr = Field(..., description="Display string for the parameter.") + variableName: StrictStr = Field( + ..., description="Python variable name of the parameter." + ) + description: Optional[StrictStr] = Field( None, description="Detailed description of the parameter." ) - suffix: Optional[str] = Field( + suffix: Optional[StrictStr] = Field( None, description="Units (like mL, mm/sec, etc) or a custom suffix for the parameter.", ) @@ -894,17 +904,17 @@ class NumberParameter(RTPBase): type: Literal["int", "float"] = Field( ..., description="String specifying whether the number is an int or float type." ) - min: float = Field( + min: Union[StrictInt, StrictFloat] = Field( ..., description="Minimum value that the number param is allowed to have." ) - max: float = Field( + max: Union[StrictInt, StrictFloat] = Field( ..., description="Maximum value that the number param is allowed to have." ) - value: float = Field( + value: Union[StrictInt, StrictFloat] = Field( ..., description="The value assigned to the parameter; if not supplied by the client, will be assigned the default value.", ) - default: float = Field( + default: Union[StrictInt, StrictFloat] = Field( ..., description="Default value of the parameter, to be used when there is no client-specified value.", ) @@ -916,11 +926,11 @@ class BooleanParameter(RTPBase): type: Literal["bool"] = Field( default="bool", description="String specifying the type of this parameter" ) - value: bool = Field( + value: StrictBool = Field( ..., description="The value assigned to the parameter; if not supplied by the client, will be assigned the default value.", ) - default: bool = Field( + default: StrictBool = Field( ..., description="Default value of the parameter, to be used when there is no client-specified value.", ) @@ -929,8 +939,10 @@ class BooleanParameter(RTPBase): class EnumChoice(BaseModel): """Components of choices used in RTP Enum Parameters.""" - displayName: str = Field(..., description="Display string for the param's choice.") - value: Union[float, str] = Field( + displayName: StrictStr = Field( + ..., description="Display string for the param's choice." + ) + value: Union[StrictInt, StrictFloat, StrictStr] = Field( ..., description="Enum value of the param's choice." ) @@ -945,11 +957,11 @@ class EnumParameter(RTPBase): choices: List[EnumChoice] = Field( ..., description="List of valid choices for this parameter." ) - value: Union[float, str] = Field( + value: Union[StrictInt, StrictFloat, StrictStr] = Field( ..., description="The value assigned to the parameter; if not supplied by the client, will be assigned the default value.", ) - default: Union[float, str] = Field( + default: Union[StrictInt, StrictFloat, StrictStr] = Field( ..., description="Default value of the parameter, to be used when there is no client-specified value.", ) @@ -958,5 +970,5 @@ class EnumParameter(RTPBase): RunTimeParameter = Union[NumberParameter, EnumParameter, BooleanParameter] RunTimeParamValuesType = Dict[ - str, Union[float, bool, str] + StrictStr, Union[StrictInt, StrictFloat, StrictBool, StrictStr] ] # update value types as more RTP types are added From 6fe994e2e73232542d4cdadf0117ee680d1937a2 Mon Sep 17 00:00:00 2001 From: Josh McVey Date: Mon, 22 Jul 2024 12:05:44 -0500 Subject: [PATCH 4/7] build: Sign windows builds via remote key storage (#15718) (#15735) ## `cherry-pick` 246efcbb52 Must update `chore_release-7.4.0` so that windows builds work and we may test an update. To understand the changes see #15718 Co-authored-by: Seth Foster --- .github/workflows/app-test-build-deploy.yaml | 77 +++++++++++++++++--- app-shell/electron-builder.config.js | 6 ++ app-shell/scripts/windows-custom-sign.js | 62 ++++++++++++++++ 3 files changed, 134 insertions(+), 11 deletions(-) create mode 100644 app-shell/scripts/windows-custom-sign.js diff --git a/.github/workflows/app-test-build-deploy.yaml b/.github/workflows/app-test-build-deploy.yaml index 878a875bdfc..4cc3754ace6 100644 --- a/.github/workflows/app-test-build-deploy.yaml +++ b/.github/workflows/app-test-build-deploy.yaml @@ -184,24 +184,45 @@ jobs: echo "both develop builds for edge" echo 'variants=["release", "internal-release"]' >> $GITHUB_OUTPUT echo 'type=develop' >> $GITHUB_OUTPUT - elif [ "${{ format('{0}', endsWith(github.ref, 'app-build-internal')) }}" = "true" ] ; then - echo "internal-release builds for app-build-internal suffixes" + elif [ "${{ format('{0}', contains(github.ref, 'app-build-internal')) }}" = "true" ] ; then + echo 'variants=["internal-release"]' >> $GITHUB_OUTPUT - echo 'type=develop' >> $GITHUB_OUTPUT - elif [ "${{ format('{0}', endsWith(github.ref, 'app-build')) }}" = "true" ] ; then - echo "release develop builds for app-build suffixes" + if [ "${{ format('{0}', contains(github.ref, 'as-release')) }}" = "true" ] ; then + echo "internal-release as-release builds for app-build-internal + as-release suffixes" + echo 'type=as-release' >> $GITHUB_OUTPUT + else + echo "internal-release develop builds for app-build-internal suffixes" + echo 'type=develop' >> $GITHUB_OUTPUT + fi + elif [ "${{ format('{0}', contains(github.ref, 'app-build')) }}" = "true" ] ; then echo 'variants=["release"]' >> $GITHUB_OUTPUT - echo 'type=develop' >> $GITHUB_OUTPUT - elif [ "${{ format('{0}', endsWith(github.ref, 'app-build-both')) }}" = "true" ] ; then - echo "Both develop builds for app-build-both suffixes" + if [ "${{ format('{0}', contains(github.ref, 'as-release')) }}" = "true" ] ; then + echo "release as-release builds for app-build + as-release suffixes" + echo 'type=as-release' >> $GITHUB_OUTPUT + else + echo "release develop builds for app-build suffixes" + echo 'type=develop' >> $GITHUB_OUTPUT + fi + elif [ "${{ format('{0}', contains(github.ref, 'app-build-both')) }}" = "true" ] ; then + echo 'variants=["release", "internal-release"]' >> $GITHUB_OUTPUT - echo 'type=develop' >> $GITHUB_OUTPUT + if [ "${{ format('{0}', contains(github.ref, 'as-release')) }}" = "true" ] ; then + echo "Both as-release builds for app-build-both + as-release suffixes" + echo 'type=as-release' >> $GITHUB_OUTPUT + else + echo "Both develop builds for app-build-both + as-release suffixes" + echo 'type=develop' >> $GITHUB_OUTPUT + fi else echo "No build for ref ${{github.ref}} and event ${{github.event_type}}" echo 'variants=[]' >> $GITHUB_OUTPUT echo 'type=develop' >> $GITHUB_OUTPUT fi + - name: set summary + run: | + echo 'Type: ${{steps.determine-build-type.outputs.type}} Variants: ${{steps.determine-build-type.outputs.variants}}' >> $GITHUB_STEP_SUMMARY + build-app: needs: [determine-build-type] if: needs.determine-build-type.outputs.variants != '[]' @@ -276,6 +297,34 @@ jobs: npm config set cache ${{ github.workspace }}/.npm-cache yarn config set cache-folder ${{ github.workspace }}/.yarn-cache make setup-js + + - name: 'Configure Windows code signing environment' + if: startsWith(matrix.os, 'windows') && contains(needs.determine-build-type.outputs.type, 'release') + shell: bash + run: | + echo "${{ secrets.SM_CLIENT_CERT_FILE_B64 }}" | base64 --decode > /d/Certificate_pkcs12.p12 + echo "${{ secrets.WINDOWS_CSC_B64}}" | base64 --decode > /d/opentrons_labworks_inc.crt + echo "C:\Program Files (x86)\Windows Kits\10\App Certification Kit" >> $GITHUB_PATH + echo "C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin\NETFX 4.8 Tools" >> $GITHUB_PATH + echo "C:\Program Files\DigiCert\DigiCert Keylocker Tools" >> $GITHUB_PATH + + - name: 'Setup Windows code signing helpers' + if: startsWith(matrix.os, 'windows') && contains(needs.determine-build-type.outputs.type, 'release') + shell: cmd + env: + SM_HOST: ${{ secrets.SM_HOST }} + SM_CLIENT_CERT_FILE: "D:\\Certificate_pkcs12.p12" + SM_CLIENT_CERT_PASSWORD: ${{secrets.SM_CLIENT_CERT_PASSWORD}} + SM_API_KEY: ${{secrets.SM_API_KEY}} + run: | + curl -X GET https://one.digicert.com/signingmanager/api-ui/v1/releases/Keylockertools-windows-x64.msi/download -H "x-api-key:${{secrets.SM_API_KEY}}" -o Keylockertools-windows-x64.msi + msiexec /i Keylockertools-windows-x64.msi /quiet /qn + smksp_registrar.exe list + smctl.exe keypair ls + C:\Windows\System32\certutil.exe -csp "DigiCert Signing Manager KSP" -key -user + smksp_cert_sync.exe + smctl.exe healthcheck --all + # build the desktop app and deploy it - name: 'build ${{matrix.variant}} app for ${{ matrix.os }}' if: matrix.target == 'desktop' @@ -283,8 +332,14 @@ jobs: env: OT_APP_MIXPANEL_ID: ${{ secrets.OT_APP_MIXPANEL_ID }} OT_APP_INTERCOM_ID: ${{ secrets.OT_APP_INTERCOM_ID }} - WIN_CSC_LINK: ${{ secrets.OT_APP_CSC_WINDOWS }} - WIN_CSC_KEY_PASSWORD: ${{ secrets.OT_APP_CSC_KEY_WINDOWS }} + WINDOWS_SIGN: ${{ format('{0}', contains(needs.determine-build-type.outputs.type, 'release')) }} + SM_HOST: ${{secrets.SM_HOST}} + SM_CLIENT_CERT_FILE: "D:\\Certificate_pkcs12.p12" + SM_CLIENT_CERT_PASSWORD: ${{secrets.SM_CLIENT_CERT_PASSWORD}} + SM_API_KEY: ${{secrets.SM_API_KEY}} + SM_CODE_SIGNING_CERT_SHA1_HASH: ${{secrets.SM_CODE_SIGNING_CERT_SHA1_HASH}} + SM_KEYPAIR_ALIAS: ${{secrets.SM_KEYPAIR_ALIAS}} + WINDOWS_CSC_FILEPATH: "D:\\opentrons_labworks_inc.crt" CSC_LINK: ${{ secrets.OT_APP_CSC_MACOS }} CSC_KEY_PASSWORD: ${{ secrets.OT_APP_CSC_KEY_MACOS }} APPLE_ID: ${{ secrets.OT_APP_APPLE_ID }} diff --git a/app-shell/electron-builder.config.js b/app-shell/electron-builder.config.js index aa61720338b..2e66f01690c 100644 --- a/app-shell/electron-builder.config.js +++ b/app-shell/electron-builder.config.js @@ -8,6 +8,7 @@ const { } = process.env const DEV_MODE = process.env.NODE_ENV !== 'production' const USE_PYTHON = process.env.NO_PYTHON !== 'true' +const WINDOWS_SIGN = process.env.WINDOWS_SIGN === 'true' const project = process.env.OPENTRONS_PROJECT ?? 'robot-stack' // this will generate either @@ -72,6 +73,11 @@ module.exports = async () => ({ target: ['nsis'], publisherName: 'Opentrons Labworks Inc.', icon: project === 'robot-stack' ? 'build/icon.ico' : 'build/three.ico', + forceCodeSigning: WINDOWS_SIGN, + rfc3161TimeStampServer: 'http://timestamp.digicert.com', + sign: 'scripts/windows-custom-sign.js', + signDlls: true, + signingHashAlgorithms: ['sha256'], }, nsis: { oneClick: false, diff --git a/app-shell/scripts/windows-custom-sign.js b/app-shell/scripts/windows-custom-sign.js new file mode 100644 index 00000000000..90d7927ab6a --- /dev/null +++ b/app-shell/scripts/windows-custom-sign.js @@ -0,0 +1,62 @@ +// from https://github.com/electron-userland/electron-builder/issues/7605 + +'use strict' + +const { execSync } = require('node:child_process') + +exports.default = async configuration => { + const signCmd = `smctl sign --keypair-alias="${String( + process.env.SM_KEYPAIR_ALIAS + )}" --input "${String(configuration.path)}" --certificate="${String( + process.env.WINDOWS_CSC_FILEPATH + )}" --exit-non-zero-on-fail --failfast --verbose` + console.log(signCmd) + try { + const signProcess = execSync(signCmd, { + stdio: 'pipe', + }) + console.log(`Sign success!`) + console.log( + `Sign stdout: ${signProcess?.stdout?.toString() ?? ''}` + ) + console.log( + `Sign stderr: ${signProcess?.stderr?.toString() ?? ''}` + ) + console.log(`Sign code: ${signProcess.code}`) + } catch (err) { + console.error(`Exception running sign: ${err.status}! +Process stdout: + ${err?.stdout?.toString() ?? ''} +------------- +Process stderr: +${err?.stdout?.toString() ?? ''} +------------- +`) + throw err + } + const verifyCmd = `smctl sign verify --fingerprint="${String( + process.env.SM_CODE_SIGNING_CERT_SHA1_HASH + )}" --input="${String(configuration.path)}" --verbose` + console.log(verifyCmd) + try { + const verifyProcess = execSync(verifyCmd, { stdio: 'pipe' }) + console.log(`Verify success!`) + console.log( + `Verify stdout: ${verifyProcess?.stdout?.toString() ?? ''}` + ) + console.log( + `Verify stderr: ${verifyProcess?.stderr?.toString() ?? ''}` + ) + } catch (err) { + console.error(` +Exception running verification: ${err.status}! +Process stdout: + ${err?.stdout?.toString() ?? ''} +-------------- +Process stderr: + ${err?.stderr?.toString() ?? ''} +-------------- +`) + throw err + } +} From 305f00ec042a55aa55f7c3096d5766f87be1c011 Mon Sep 17 00:00:00 2001 From: Josh McVey Date: Mon, 22 Jul 2024 14:24:38 -0500 Subject: [PATCH 5/7] fix(scripts): setuptools v71 removed the need for extern (#15717) (#15741) ## `cherry-pick` afaa7177fe Must update `chore_release-7.4.0` so that Linux app builds work due to a change in setuptools. To understand the changes see #15717 Co-authored-by: Ryan Howard --- scripts/python_build_utils.py | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/scripts/python_build_utils.py b/scripts/python_build_utils.py index d55ece0e0c8..eff17678fda 100644 --- a/scripts/python_build_utils.py +++ b/scripts/python_build_utils.py @@ -54,9 +54,18 @@ def normalize_version(package, project, extra_tag='', git_dir=None): # the way they vendor dependencies, like the packaging module that # provides the way to normalize version numbers for wheel file names. So # we try all the possible ways to find it. + # Since 71.0.0 they have removed the need for extern + # So depending on the version of 3.10 you're building on you may or may not + # need to use the extern or import it directly try: - # new way - from setuptools.extern import packaging + import setuptools + major, minor, patch = [int(x, 10) for x in setuptools.__version__.split('.')] + if major < 71: + # new way + from setuptools.extern import packaging + else: + # new new way + import packaging except ImportError: # old way from pkg_resources.extern import packaging From 043bca7aec2a0279d91b8e5b81273454e6dfbabb Mon Sep 17 00:00:00 2001 From: Shlok Amin Date: Mon, 1 Jul 2024 14:43:17 -0400 Subject: [PATCH 6/7] ci(app-shell): update macos github runner version (#15559) --- .github/workflows/app-test-build-deploy.yaml | 6 +++--- app-shell/Makefile | 3 --- 2 files changed, 3 insertions(+), 6 deletions(-) diff --git a/.github/workflows/app-test-build-deploy.yaml b/.github/workflows/app-test-build-deploy.yaml index 4cc3754ace6..79a234eba14 100644 --- a/.github/workflows/app-test-build-deploy.yaml +++ b/.github/workflows/app-test-build-deploy.yaml @@ -96,7 +96,7 @@ jobs: backend-unit-test: strategy: matrix: - os: ['windows-2022', 'ubuntu-22.04', 'macos-11'] + os: ['windows-2022', 'ubuntu-22.04', 'macos-latest'] name: 'opentrons app backend unit tests on ${{matrix.os}}' runs-on: ${{ matrix.os }} steps: @@ -228,13 +228,13 @@ jobs: if: needs.determine-build-type.outputs.variants != '[]' strategy: matrix: - os: ['windows-2022', 'ubuntu-22.04', 'macos-11'] + os: ['windows-2022', 'ubuntu-22.04', 'macos-latest'] variant: ${{fromJSON(needs.determine-build-type.outputs.variants)}} target: ['desktop', 'odd'] exclude: - os: 'windows-2022' target: 'odd' - - os: 'macos-11' + - os: 'macos-latest' target: 'odd' runs-on: ${{ matrix.os }} diff --git a/app-shell/Makefile b/app-shell/Makefile index ec86ee924ff..32c8b7ce01e 100644 --- a/app-shell/Makefile +++ b/app-shell/Makefile @@ -145,9 +145,6 @@ dist-ot3: package-deps .PHONY: dist-macos-latest dist-macos-latest: dist-osx -.PHONY: dist-macos-11 -dist-macos-11: dist-osx - .PHONY: dist-ubuntu-latest dist-ubuntu-latest: dist-linux From a77ffd7a091e3c1a603ea0bcf9c67ecda94bb9a5 Mon Sep 17 00:00:00 2001 From: Josh McVey Date: Thu, 11 Jul 2024 14:32:05 -0500 Subject: [PATCH 7/7] RQA-2830 fix mac dmg to be x64 binary and use x64 bundled python to pip install (#15624) ## Replaces #15618 with PR against branch name ending in `app-build-internal` to trigger a build ## Overview - - `macos-11` GitHub runner was intel and now we use macos-latest that is arm [doc](https://docs.github.com/en/actions/using-github-hosted-runners/about-github-hosted-runners/about-github-hosted-runners#standard-github-hosted-runners-for-public-repositories) - specify --x64 argument to electron-builder - ~~per [doc](https://github.com/actions/setup-python?tab=readme-ov-file#supported-architectures) x64 python should be installed so the pip install will be the right platform. I'm not confident in this.~~ - see comment in #15618 - we need to use the python we download on mac to do the pip install so the installed packages match the platform correctly ## Test Plan - [x] build locally and see that it works - [x] watch CI and install the built package - [x] validate that once the package dmg is x64 that analysis works (python has packages with the right arch) # Review requests - This the right approach? # Risk assessment High because our .dmg currently cannot analyze protocols 2.0.0-alpha.0 internal release build. --- app-shell/Makefile | 2 +- app-shell/scripts/before-pack.js | 11 ++++++++++- 2 files changed, 11 insertions(+), 2 deletions(-) diff --git a/app-shell/Makefile b/app-shell/Makefile index 32c8b7ce01e..9624aa1a54e 100644 --- a/app-shell/Makefile +++ b/app-shell/Makefile @@ -122,7 +122,7 @@ dist-posix: package-deps .PHONY: dist-osx dist-osx: package-deps - $(builder) --mac + $(builder) --mac --x64 $(MAKE) _dist-collect-artifacts .PHONY: dist-linux diff --git a/app-shell/scripts/before-pack.js b/app-shell/scripts/before-pack.js index 6e96609786f..bb5e2d45485 100644 --- a/app-shell/scripts/before-pack.js +++ b/app-shell/scripts/before-pack.js @@ -118,7 +118,16 @@ module.exports = function beforeBuild(context) { // TODO(mc, 2022-05-16): explore virtualenvs for a more reliable // implementation of this install - return execa(HOST_PYTHON, [ + console.log( + `Installing python native deps using ${path.join( + PYTHON_DESTINATION, + 'python3.10' + )}` + ) + const invokablePython = platformName.includes('darwin') + ? path.join(PYTHON_DESTINATION, 'python/bin/python3.10') + : HOST_PYTHON + return execa(invokablePython, [ '-m', 'pip', 'install',