Skip to content

Commit

Permalink
Merge branch 'release-0.2.0' into chore/alex-update-test-users-241211
Browse files Browse the repository at this point in the history
  • Loading branch information
AlexZorkin authored Dec 11, 2024
2 parents 3e90bbc + aa0c65d commit cacb11c
Show file tree
Hide file tree
Showing 60 changed files with 578 additions and 358 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
"""Update Fuel Types measured in volume to be other-uses
Revision ID: 7ae38a8413ab
Revises: 26ab15f8ab18
Create Date: 2024-12-09 19:31:18.199089
"""

import sqlalchemy as sa
from alembic import op
from datetime import datetime

# revision identifiers, used by Alembic.
revision = "7ae38a8413ab"
down_revision = "cd8698fe40e6"
branch_labels = None
depends_on = None


def upgrade() -> None:
current_time = datetime.now()

# Update the `other_uses_fossil_derived` field for all specified fuel types
op.execute(
f"""
UPDATE fuel_type
SET other_uses_fossil_derived = true,
update_date = '{current_time}',
update_user = 'no_user'
WHERE fuel_type IN (
'Alternative jet fuel',
'Biodiesel',
'Ethanol',
'HDRD',
'Renewable gasoline',
'Renewable naphtha'
)
"""
)

# Update the `other_uses_fossil_derived` field for all specified fuel types
op.execute(
f"""
UPDATE fuel_type
SET other_uses_fossil_derived = false,
update_date = '{current_time}',
update_user = 'no_user'
WHERE fuel_type IN (
'CNG',
'Electricity',
'Hydrogen',
'LNG',
'Propane'
)
"""
)


def downgrade() -> None:
current_time = datetime.now()

# Revert the `other_uses_fossil_derived` field to false for the first set of fuel types
op.execute(
f"""
UPDATE fuel_type
SET other_uses_fossil_derived = false,
update_date = '{current_time}',
update_user = 'no_user'
WHERE fuel_type IN (
'Alternative jet fuel',
'Biodiesel',
'Ethanol',
'HDRD',
'Renewable gasoline',
'Renewable naphtha'
)
"""
)

# Revert the `other_uses_fossil_derived` field to true for the second set of fuel types
op.execute(
f"""
UPDATE fuel_type
SET other_uses_fossil_derived = true,
update_date = '{current_time}',
update_user = 'no_user'
WHERE fuel_type IN (
'CNG',
'Electricity',
'Hydrogen',
'LNG',
'Propane'
)
"""
)
55 changes: 36 additions & 19 deletions backend/lcfs/tests/other_uses/test_other_uses_repo.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,14 +11,19 @@


@pytest.fixture
def mock_db_session():
def mock_query_result():
# Setup mock for database query result chain
mock_result = AsyncMock()
mock_result.unique = MagicMock(return_value=mock_result)
mock_result.scalars = MagicMock(return_value=mock_result)
mock_result.all = MagicMock(return_value=[MagicMock(spec=OtherUses)])
return mock_result


@pytest.fixture
def mock_db_session(mock_query_result):
session = MagicMock(spec=AsyncSession)
execute_result = AsyncMock()
execute_result.unique = MagicMock(return_value=execute_result)
execute_result.scalars = MagicMock(return_value=execute_result)
execute_result.all = MagicMock(return_value=[MagicMock(spec=OtherUses)])
execute_result.first = MagicMock(return_value=MagicMock(spec=OtherUses))
session.execute.return_value = execute_result
session.execute = AsyncMock(return_value=mock_query_result)
return session


Expand All @@ -29,6 +34,14 @@ def other_uses_repo(mock_db_session):
repo.fuel_code_repo.get_fuel_categories = AsyncMock(return_value=[])
repo.fuel_code_repo.get_fuel_types = AsyncMock(return_value=[])
repo.fuel_code_repo.get_expected_use_types = AsyncMock(return_value=[])

# Mock for local get_formatted_fuel_types method
async def mock_get_formatted_fuel_types():
mock_result = await mock_db_session.execute(AsyncMock())
return mock_result.unique().scalars().all()

repo.get_formatted_fuel_types = AsyncMock(side_effect=mock_get_formatted_fuel_types)

return repo


Expand Down Expand Up @@ -194,22 +207,21 @@ async def test_get_latest_other_uses_by_group_uuid(other_uses_repo, mock_db_sess
mock_other_use_gov.user_type = UserTypeEnum.GOVERNMENT
mock_other_use_gov.version = 2

mock_other_use_supplier = MagicMock(spec=OtherUses)
mock_other_use_supplier.user_type = UserTypeEnum.SUPPLIER
mock_other_use_supplier.version = 3
# Setup mock result chain
mock_result = AsyncMock()
mock_result.unique = MagicMock(return_value=mock_result)
mock_result.scalars = MagicMock(return_value=mock_result)
mock_result.first = MagicMock(return_value=mock_other_use_gov)

# Mock response with both government and supplier versions
mock_db_session.execute.return_value.scalars.return_value.first.side_effect = [
mock_other_use_gov,
mock_other_use_supplier,
]
# Configure mock db session
mock_db_session.execute = AsyncMock(return_value=mock_result)
other_uses_repo.db = mock_db_session

result = await other_uses_repo.get_latest_other_uses_by_group_uuid(group_uuid)

assert result.user_type == UserTypeEnum.GOVERNMENT
assert result.version == 2


@pytest.mark.anyio
async def test_get_other_use_version_by_user(other_uses_repo, mock_db_session):
group_uuid = "test-group-uuid"
Expand All @@ -221,9 +233,14 @@ async def test_get_other_use_version_by_user(other_uses_repo, mock_db_session):
mock_other_use.version = version
mock_other_use.user_type = user_type

mock_db_session.execute.return_value.scalars.return_value.first.return_value = (
mock_other_use
)
# Set up mock result chain
mock_result = AsyncMock()
mock_result.scalars = MagicMock(return_value=mock_result)
mock_result.first = MagicMock(return_value=mock_other_use)

# Configure mock db session
mock_db_session.execute = AsyncMock(return_value=mock_result)
other_uses_repo.db = mock_db_session

result = await other_uses_repo.get_other_use_version_by_user(
group_uuid, version, user_type
Expand Down
97 changes: 90 additions & 7 deletions backend/lcfs/web/api/other_uses/repo.py
Original file line number Diff line number Diff line change
@@ -1,20 +1,21 @@
import structlog
from typing import List, Optional, Tuple, Any
from datetime import date
from typing import List, Optional, Tuple, Dict, Any

from fastapi import Depends

from lcfs.db.base import ActionTypeEnum, UserTypeEnum
from lcfs.db.dependencies import get_async_db_session

from sqlalchemy import select, delete, func, case, and_
from sqlalchemy.orm import joinedload
from sqlalchemy import select, delete, func, case, and_, or_
from sqlalchemy.orm import joinedload, contains_eager
from sqlalchemy.ext.asyncio import AsyncSession

from lcfs.db.models.compliance import ComplianceReport
from lcfs.db.models.compliance.OtherUses import OtherUses
from lcfs.db.models.fuel.ProvisionOfTheAct import ProvisionOfTheAct
from lcfs.db.models.fuel.FuelCode import FuelCode
from lcfs.db.models.fuel.FuelType import QuantityUnitsEnum
from lcfs.db.models.fuel.FuelType import FuelType, QuantityUnitsEnum
from lcfs.db.models.fuel.FuelInstance import FuelInstance
from lcfs.web.api.fuel_code.repo import FuelCodeRepository
from lcfs.web.api.other_uses.schema import OtherUsesSchema
from lcfs.web.api.base import PaginationRequestSchema
Expand All @@ -37,7 +38,7 @@ def __init__(
async def get_table_options(self) -> dict:
"""Get all table options"""
fuel_categories = await self.fuel_code_repo.get_fuel_categories()
fuel_types = await self.fuel_code_repo.get_formatted_fuel_types()
fuel_types = await self.get_formatted_fuel_types()
expected_uses = await self.fuel_code_repo.get_expected_use_types()
units_of_measure = [unit.value for unit in QuantityUnitsEnum]
provisions_of_the_act = (
Expand Down Expand Up @@ -75,7 +76,7 @@ async def get_latest_other_uses_by_group_uuid(
)

result = await self.db.execute(query)
return result.scalars().first()
return result.unique().scalars().first()

@repo_handler
async def get_other_uses(self, compliance_report_id: int) -> List[OtherUsesSchema]:
Expand Down Expand Up @@ -302,3 +303,85 @@ async def get_other_use_version_by_user(

result = await self.db.execute(query)
return result.scalars().first()

@repo_handler
async def get_formatted_fuel_types(self) -> List[Dict[str, Any]]:
"""Get all fuel type options with their associated fuel categories and fuel codes for other uses"""
# Define the filtering conditions for fuel codes
current_date = date.today()
fuel_code_filters = (
or_(
FuelCode.effective_date == None, FuelCode.effective_date <= current_date
)
& or_(
FuelCode.expiration_date == None,
FuelCode.expiration_date > current_date,
)
& (FuelType.other_uses_fossil_derived == True)
)

# Build the query with filtered fuel_codes
query = (
select(FuelType)
.outerjoin(FuelType.fuel_instances)
.outerjoin(FuelInstance.fuel_category)
.outerjoin(FuelType.fuel_codes)
.where(fuel_code_filters)
.options(
contains_eager(FuelType.fuel_instances).contains_eager(
FuelInstance.fuel_category
),
contains_eager(FuelType.fuel_codes),
joinedload(FuelType.provision_1),
joinedload(FuelType.provision_2),
)
)

result = await self.db.execute(query)
fuel_types = result.unique().scalars().all()

# Prepare the data in the format matching your schema
formatted_fuel_types = []
for fuel_type in fuel_types:
formatted_fuel_type = {
"fuel_type_id": fuel_type.fuel_type_id,
"fuel_type": fuel_type.fuel_type,
"default_carbon_intensity": fuel_type.default_carbon_intensity,
"units": fuel_type.units if fuel_type.units else None,
"unrecognized": fuel_type.unrecognized,
"fuel_categories": [
{
"fuel_category_id": fc.fuel_category.fuel_category_id,
"category": fc.fuel_category.category,
}
for fc in fuel_type.fuel_instances
],
"fuel_codes": [
{
"fuel_code_id": fc.fuel_code_id,
"fuel_code": fc.fuel_code,
"carbon_intensity": fc.carbon_intensity,
}
for fc in fuel_type.fuel_codes
],
"provision_of_the_act": [],
}

if fuel_type.provision_1:
formatted_fuel_type["provision_of_the_act"].append(
{
"provision_of_the_act_id": fuel_type.provision_1_id,
"name": fuel_type.provision_1.name,
}
)

if fuel_type.provision_2:
formatted_fuel_type["provision_of_the_act"].append(
{
"provision_of_the_act_id": fuel_type.provision_2_id,
"name": fuel_type.provision_2.name,
}
)
formatted_fuel_types.append(formatted_fuel_type)

return formatted_fuel_types
8 changes: 4 additions & 4 deletions frontend/src/components/BCForm/BCFormCheckbox.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,9 @@ import {
Checkbox,
FormControl,
FormControlLabel,
FormLabel,
Typography
FormLabel
} from '@mui/material'
import BCTypography from '@/components/BCTypography'
import { Controller } from 'react-hook-form'
import { CustomLabel } from './CustomLabel'
import PropTypes from 'prop-types'
Expand All @@ -22,9 +22,9 @@ export const BCFormCheckbox = ({ name, form, label, options, disabled }) => {
return (
<FormControl size={'small'} variant={'outlined'}>
<FormLabel component="legend">
<Typography variant="label" component="span">
<BCTypography variant="label" component="span">
{label}
</Typography>
</BCTypography>
</FormLabel>

<div>
Expand Down
8 changes: 4 additions & 4 deletions frontend/src/components/BCForm/BCFormRadio.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,9 @@ import {
FormControlLabel,
FormLabel,
Radio,
RadioGroup,
Typography
RadioGroup
} from '@mui/material'
import BCTypography from '@/components/BCTypography'
import { Controller } from 'react-hook-form'
import PropTypes from 'prop-types'
import { CustomLabel } from './CustomLabel'
Expand Down Expand Up @@ -34,9 +34,9 @@ export const BCFormRadio = ({ name, control, label, options, disabled }) => {
return (
<FormControl component="fieldset">
<FormLabel component="legend" sx={{ marginBottom: 1 }}>
<Typography variant="label" component="span">
<BCTypography variant="label" component="span">
{label}
</Typography>
</BCTypography>
</FormLabel>
<Controller
name={name}
Expand Down
7 changes: 4 additions & 3 deletions frontend/src/components/BCForm/BCFormText.jsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { Controller } from 'react-hook-form'
import { TextField, InputLabel, Typography } from '@mui/material'
import { TextField, InputLabel } from '@mui/material'
import BCTypography from '@/components/BCTypography'
import PropTypes from 'prop-types'

export const BCFormText = ({ name, control, label, optional }) => {
Expand All @@ -14,14 +15,14 @@ export const BCFormText = ({ name, control, label, optional }) => {
}) => (
<>
<InputLabel htmlFor={name} component="label" className="form-label">
<Typography variant="label" component="span">
<BCTypography variant="label" component="span">
{label}&nbsp;
{optional && (
<span className="optional" style={{ fontWeight: 'normal' }}>
(optional)
</span>
)}
</Typography>
</BCTypography>
</InputLabel>
<TextField
id={name}
Expand Down
Loading

0 comments on commit cacb11c

Please sign in to comment.