diff --git a/catalogs/cpu_table.tsv b/catalogs/cpu_table.tsv index 20d582c..91ef060 100644 --- a/catalogs/cpu_table.tsv +++ b/catalogs/cpu_table.tsv @@ -194,6 +194,44 @@ intel-xeon-platinum-8362 2.8 32 2 265 0 0 intel-xeon-platinum-8368 2.4 38 2 270 0 0 intel-xeon-platinum-8380 2.3 40 2 270 0 0 +intel-xeon-bronze-3408u 1.8 8 2 125 0 0 +intel-xeon-silver-4410y 2.0 12 2 150 0 0 +intel-xeon-silver-4416+ 2.0 20 2 165 0 0 +intel-xeon-gold-5411n 1.9 24 2 165 0 0 +intel-xeon-gold-5415+ 2.9 8 2 150 0 0 +intel-xeon-gold-5416s 2.0 16 2 150 0 0 +intel-xeon-gold-5418n 1.8 24 2 165 0 0 +intel-xeon-gold-5418y 2.0 24 2 185 0 0 +intel-xeon-gold-5420+ 2.0 28 2 205 0 0 +intel-xeon-gold-6414u 2.0 32 1 250 0 0 +intel-xeon-gold-6416h 2.2 18 2 165 0 0 +intel-xeon-gold-6418h 2.1 24 2 185 0 0 +intel-xeon-gold-6421n 1.8 32 2 185 0 0 +intel-xeon-gold-6426y 2.5 16 2 185 0 0 +intel-xeon-gold-6430 2.1 32 2 270 0 0 +intel-xeon-gold-6434 3.7 8 2 195 0 0 +intel-xeon-gold-6438n 2.0 32 2 205 0 0 +intel-xeon-gold-6438y+ 2.0 32 2 205 0 0 +intel-xeon-gold-6442y 2.6 24 2 225 0 0 +intel-xeon-gold-6444y 3.6 16 2 270 0 0 +intel-xeon-gold-6448y 2.1 32 2 225 0 0 +intel-xeon-gold-6454s 2.2 32 2 270 0 0 +intel-xeon-gold-6458q 3.1 32 2 350 0 0 +intel-xeon-platinum-8444h 2.9 16 2 270 0 0 +intel-xeon-platinum-8452y 2.0 36 2 300 0 0 +intel-xeon-platinum-8458p 2.7 44 2 350 0 0 +intel-xeon-platinum-8458q 3.1 32 2 350 0 0 +intel-xeon-platinum-8460y+ 2.0 40 2 300 0 0 +intel-xeon-platinum-8462y+ 2.8 32 2 300 0 0 +intel-xeon-platinum-8468 2.1 48 2 350 0 0 +intel-xeon-platinum-8468v 2.4 48 2 330 0 0 +intel-xeon-platinum-8470 2.0 52 2 350 0 0 +intel-xeon-platinum-8470n 1.7 52 2 300 0 0 +intel-xeon-platinum-8470q 2.1 52 2 350 0 0 +intel-xeon-platinum-8480+ 2.0 56 2 350 0 0 +intel-xeon-platinum-8490h 1.9 60 2 350 0 0 +intel-xeon-platinum-9462 2.7 32 2 350 0 0 + amd-epyc-7262 3.2 8 2 155 0 0 amd-epyc-7272 2.9 12 2 120 0 0 amd-epyc-7282 2.8 16 2 120 0 0 diff --git a/concho/config.py b/concho/config.py index 0f854ec..3619ea4 100644 --- a/concho/config.py +++ b/concho/config.py @@ -43,17 +43,50 @@ class Chassis(Item): self.num_dimm_slots_per_channel = 2 +class SdramChip(): + # eg 'sdr66', 'ddr-400', 'ddr4-2666', 'ddr5-4800', see https://en.wikipedia.org/wiki/DIMM + chip_type: str # 'sdr' or 'ddr' (single or dual data rate) + generation: Optional[int] + transfer_rate: int # in mega transfer per second + + def __init__(self, chip_type: str, generation: Optional[int], transfer_rate: int): + self.chip_type = chip_type + self.generation = generation + self.transfer_rate = transfer_rate + + +class DimmCas(): + # https://en.wikipedia.org/wiki/CAS_latency + cas1: int + cas2: int + cas3: int + + def __init__(self, cas1: int, cas2: int, cas3: int): + self.cas1 = cas1 + self.cas2 = cas2 + self.cas3 = cas3 + + class Dimm(Item): num_gb: int - num_mhz: int + sdram_chip: SdramChip mem_type: MemType + cas: Optional[DimmCas] - def __init__(self, num_gb, num_mhz, mem_type): - uid = "%s-%s-%s" % (mem_type, num_gb, num_mhz) - super().__init__(uid) + def __init__(self, num_gb: int, sdram_chip: SdramChip, cas: Optional[DimmCas], mem_type: str): + ''' + mem_type: 'rdimm', 'pmm' + ''' self.num_gb = num_gb - self.num_mhz = num_mhz + self.sdram_chip = sdram_chip + self.cas = cas self.mem_type = mem_type + uid = "%s-%s-%s" % (mem_type, num_gb, self.num_mhz) + super().__init__(uid) + + @property + def num_mhz(self): + return self.sdram_chip.transfer_rate class Cpu(Item): @@ -83,6 +116,14 @@ class Cpu(Item): proc_id = self.uid if re.match('intel-core-i[357]-8[0-9][0-9][0-9][ktbuh]', proc_id): return 'coffeelake' + elif re.match('intel-xeon-bronze-[0-9]4[0-9][0-9]', proc_id): + return 'sapphire rapids' + elif re.match('intel-xeon-silver-[0-9]4[0-9][0-9]', proc_id): + return 'sapphire rapids' + elif re.match('intel-xeon-gold-[0-9]4[0-9][0-9]', proc_id): + return 'sapphire rapids' + elif re.match('intel-xeon-platinum-[0-9]4[0-9][0-9]', proc_id): + return 'icelake' elif re.match('intel-xeon-silver-[0-9]3[0-9][0-9]', proc_id): return 'icelake' elif re.match('intel-xeon-gold-[0-9]3[0-9][0-9]', proc_id): @@ -163,6 +204,10 @@ class Cpu(Item): # https://www.intel.com/content/www/us/en/products/sku/215269/intel-xeon-silver-4314-processor-24m-cache-2-40-ghz/specifications.html shows that even xeon silver 4314 has 2 AVX 512 fma units num_simd_per_core = 2 + if proc_arch in ['sapphire rapids']: + num_simd_per_core = 2 + # cpus_may2023_v3.pdf + if proc_arch == 'rome': num_simd_per_core = 1 @@ -177,6 +222,7 @@ class Cpu(Item): 'coffeelake': 6, 'cascadelake': 6, 'icelake': 8, + 'sapphire rapids': 8, 'rome': 8, 'milan': 8 }[self.architecture] @@ -252,6 +298,7 @@ def get_simd_id(proc_arch: CpuArchitecture) -> SimdId: 'skylake': 'avx-512', 'cascadelake': 'avx-512', 'icelake': 'avx-512', + 'sapphire rapids': 'avx-512', 'coffeelake': 'avx2', # from https://www.microway.com/knowledge-center-articles/detailed-specifications-of-the-amd-epyc-rome-cpus/: # - Full support for 256-bit AVX2 instructions with two 256-bit FMA units per CPU core. The previous “Naples” architecture split 256-bit instructions into two separate 128-bit operations @@ -308,7 +355,7 @@ class Config(): cpu: Optional[Cpu] cpu_slots_mem: List[CpuSlotMem] - def __init__(self, configurator): + def __init__(self, configurator: 'Configurator'): self.configurator = configurator self.num_servers = 0 self._num_cpu_per_server = 0 @@ -321,7 +368,7 @@ class Config(): @staticmethod def _find_dimm_combination(num_dimm_slots_per_channel: int, min_ram_per_channel: int, available_dimms: List['Option']) -> List[Dimm]: - available_dimms.append(Option(Dimm(0, 0, 'dummy'), 0.0)) # fake dimm to represent empty slot + available_dimms.append(Option(Dimm(num_gb=0, sdram_chip=SdramChip('ddr', generation=None, transfer_rate=66), cas=None, mem_type='dummy'), 0.0)) # fake dimm to represent empty slot slot_options = [] # try all combinations of dimms @@ -446,6 +493,8 @@ class Config(): dynamic_frequency_scaling = xeon_6248_avx512_base_freq / xeon_6248_base_freq elif self.cpu.architecture == 'icelake': dynamic_frequency_scaling = 0.9 # 0.9 is a guesstimate based on a web page that I found in january 2023 (I can't find it anymore) which showed that the frequence was less lowered on ice lake... if we could find actual figures, it would be great + elif self.cpu.architecture == 'saphhire rapids': + dynamic_frequency_scaling = 1.0 # sapphire rapids seem to get closer to theoretical speed, see /home/graffy/work/concho/cpus_may2023_v3.pdf cpu_clock_when_computing = self.cpu.clock * dynamic_frequency_scaling flops = self.cpu.num_dp_flop_per_cycle * cpu_clock_when_computing * 1.e9 * self.cpu.num_cores * self.num_cpu_per_server * self.num_servers return flops diff --git a/concho/dell.py b/concho/dell.py index 706ba98..a572268 100644 --- a/concho/dell.py +++ b/concho/dell.py @@ -1,3 +1,4 @@ +from typing import List from pathlib import Path from concho.config import TableBasedConfigurator from concho.config import Configurator @@ -5,12 +6,13 @@ from concho.config import Module from concho.config import Option from concho.config import Config from concho.config import Chassis -from concho.config import Cpu, Dimm +from concho.config import Cpu, Dimm, Price, MemSizeGb, MemSize, CpuId from concho.config import IHtmlConfiguratorParser from abc import abstractmethod # from xml.dom.minidom import parse -from lxml.html import parse +from lxml.html import parse as html_parse +from lxml.html import HtmlElement import re import copy @@ -242,7 +244,7 @@ class DellPowerEdgeR940(TableBasedConfigurator): def __init__(self): super().__init__('r940', num_cpu_per_server=4, num_servers=1) - def get_empty_price(self): + def get_empty_price(self) -> Price: # price of r940 (with 2x xeon gold 5215 and 32 Go DDR4 @ 2933GHz) on 09/06/2020 : 3784€ # (x: price without procs, p5215: price of gold-5215, p6248: price of Gold6248) # p6240 = 2684 @@ -255,7 +257,7 @@ class DellPowerEdgeR940(TableBasedConfigurator): # => p5215 = 1317 (agrees with proc price on r640) return 1150.0 - def get_dimm_price(self, dimm_capacity): + def get_dimm_price(self, dimm_capacity: MemSizeGb) -> Price: return { 8: 80.0, 16: 160.0, @@ -263,7 +265,7 @@ class DellPowerEdgeR940(TableBasedConfigurator): 64: 640.0 }[dimm_capacity] - def get_guarantee_price(self, guarantee_duration): + def get_guarantee_price(self, guarantee_duration: int) -> Price: if guarantee_duration > 7: assert False, 'guarantee of more than 7 years is not available on %s' % self.host_type_id elif guarantee_duration >= 5: @@ -273,7 +275,7 @@ class DellPowerEdgeR940(TableBasedConfigurator): return 0.0 @abstractmethod - def get_disk_upgrade_price(self, asked_disk_capacity): + def get_disk_upgrade_price(self, asked_disk_capacity: MemSize) -> Price: assert 1.9e12 < asked_disk_capacity < 2.1e12, 'only 2To upgrades are handled for %s' % self.host_type_id # Retrait des disques de base (2x600Go 10K SAS 2.5'') : -260.0 € # Ajout d'un disque dur 1,2 To SAS 10k Tpm 2,5" - hotplug : 165.0 € @@ -287,21 +289,21 @@ class DellPrecision3630(TableBasedConfigurator): def __init__(self): super().__init__('prceision3630', num_cpu_per_server=1, num_servers=1) - def get_empty_price(self): + def get_empty_price(self) -> Price: return 449.0 - def get_dimm_price(self, dimm_capacity): + def get_dimm_price(self, dimm_capacity: MemSizeGb) -> Price: return { 8: 80.0, 16: 160.0, 32: 320.0 }[dimm_capacity] - def get_guarantee_price(self, guarantee_duration): + def get_guarantee_price(self, guarantee_duration) -> Price: assert guarantee_duration <= 5, 'only 5 year guarantee is handled for %s' % self.host_type_id return 0.0 - def get_disk_upgrade_price(self, asked_disk_capacity): + def get_disk_upgrade_price(self, asked_disk_capacity) -> Price: assert 1.9e12 < asked_disk_capacity < 2.1e12, 'only 2To upgrades are handled for %s' % self.host_type_id return 0.0 @@ -311,7 +313,7 @@ class DellPowerEdgeC6420(TableBasedConfigurator): def __init__(self, host_type_id): super().__init__(host_type_id, num_cpu_per_server=2, num_servers=4) - def get_empty_price(self): + def get_empty_price(self) -> Price: # for 4xc6420 on 19/06/2020 (from excel quotation) # # @@ -359,7 +361,7 @@ class DellPowerEdgeC6420(TableBasedConfigurator): poweredge_c6000_price = basic_config_price - (xeon_silver_4210r_price * num_cpu_per_server + ram_price_per_gigabyte * 48) * num_servers_per_c6000 return poweredge_c6000_price - def get_dimm_price(self, dimm_capacity): + def get_dimm_price(self, dimm_capacity) -> Price: return { 8: 80.0, 16: 160.0, @@ -367,7 +369,7 @@ class DellPowerEdgeC6420(TableBasedConfigurator): 64: 640.0 }[dimm_capacity] - def get_guarantee_price(self, guarantee_duration): + def get_guarantee_price(self, guarantee_duration) -> Price: if guarantee_duration > 7: assert False, 'guarantee of more than 7 years is not available on %s' % self.host_type_id elif guarantee_duration >= 5: @@ -376,7 +378,7 @@ class DellPowerEdgeC6420(TableBasedConfigurator): # 5-year guarantee included in base price return 0.0 - def get_disk_upgrade_price(self, asked_disk_capacity): + def get_disk_upgrade_price(self, asked_disk_capacity) -> Price: assert 1.9e12 < asked_disk_capacity < 2.1e12, 'only 2To upgrades are handled for %s' % self.host_type_id # from c6420-20200716-price # | Ajout d'un disque dur 1 To SATA 7200 Tpm 3,5'' pour les 4 serveurs | | 4-3-1-14g096 | C6420 | 361 € | | € - | @@ -388,7 +390,7 @@ class DellConfiguratorParser(IHtmlConfiguratorParser): def __init__(self): pass - def _get_module(self, root_element, section_label): + def _get_module(self, root_element: HtmlElement, section_label: str) -> HtmlElement: modules_element = root_element.xpath(self.get_xpath_filter('root_to_modules_element'))[0] # print(modules_element) for module_root in modules_element.xpath(self.get_xpath_filter('modules_element_to_modules')): @@ -426,10 +428,10 @@ class DellConfiguratorParser(IHtmlConfiguratorParser): assert False @abstractmethod - def get_base_price(self, html_root): + def get_base_price(self, html_root: HtmlElement) -> Price: assert False - def _parse_proc_change_options(self, html_root): + def _parse_proc_change_options(self, html_root: HtmlElement) -> Module: proc_options = Module('processor-change') # module_root_element = self._get_module(html_root, 'Processeurs (Passage)') module_root_element = self._get_module(html_root, self.get_module_label('cpu_change')) @@ -469,7 +471,7 @@ class DellConfiguratorParser(IHtmlConfiguratorParser): proc_options.add_option(option) return proc_options - def _parse_proc_options(self, html_root): + def _parse_proc_options(self, html_root: HtmlElement) -> Module: proc_options = Module('processor') # module_root_element = self._get_module(html_root, 'Processeurs (Passage)') module_root_element = self._get_module(html_root, self.get_module_label('additional_cpus')) @@ -498,7 +500,7 @@ class DellConfiguratorParser(IHtmlConfiguratorParser): assert len(proc_options.options) > 0 return proc_options - def _parse_ram_options(self, html_root): + def _parse_ram_options(self, html_root: HtmlElement) -> Module: ram_options = Module('ram') # module_root_element = self._get_module(html_root, 'Processeurs (Passage)') module_root_element = self._get_module(html_root, self.get_module_label('ram_additions')) @@ -535,10 +537,10 @@ class DellConfiguratorParser(IHtmlConfiguratorParser): return ram_options @abstractmethod - def _get_module_default_item(self, module_label, html_root): + def _get_module_default_item(self, module_label, html_root: HtmlElement): assert False - def _parse_base_config(self, html_root, configurator): + def _parse_base_config(self, html_root: HtmlElement, configurator: Configurator) -> Config: base_config = Config(configurator) base_config.num_servers = configurator.chassis.item.max_num_servers base_config.num_cpu_per_server = 1 @@ -596,7 +598,7 @@ class DellConfiguratorParser(IHtmlConfiguratorParser): return base_config @staticmethod - def _deduce_base_cpu_price(base_cpu, cpu_options, additional_cpu_options): + def _deduce_base_cpu_price(base_cpu: CpuId, cpu_options: List[Option], additional_cpu_options: List[Option]) -> Price: ''' The price of the base config processor is not always available directly in the section 'additional processors' : as an example, r940's default processors are 2 xeon gold 5215 but it's not possible to add 2 other 5215 (probably because 5215 can only talk to another cpu, not 3). In this case the price of this base cpu can be deduced from the price for other cpus (difference between cpu upgrade and additional cpu) @@ -624,9 +626,9 @@ class DellConfiguratorParser(IHtmlConfiguratorParser): return base_cpu_price - def parse(self, dell_configurator_html_file_path, configurator): + def parse(self, dell_configurator_html_file_path: Path, configurator: Configurator): - html_root = parse(dell_configurator_html_file_path).getroot() + html_root = html_parse(dell_configurator_html_file_path).getroot() # print(dir(html_root)) # for e in html_root: # print(e.tag) diff --git a/concho/hpev2.py b/concho/hpev2.py new file mode 100644 index 0000000..fa20f99 --- /dev/null +++ b/concho/hpev2.py @@ -0,0 +1,294 @@ +from typing import List, Tuple +from concho.config import IHtmlConfiguratorParser, Configurator, Module, Option, Cpu, Price, Dimm, SdramChip, DimmCas, Config, Chassis +from pathlib import Path +from lxml.html import HtmlElement, parse as parse_html +import re +import pandas as pd +import json + + +def parse_price(price_as_str: str) -> Price: + # 'EUR 1,092.65' + return Price(price_as_str.replace('EUR', '').replace(',', '')) + + +class Quantity(): + num_selected: int # selected quantity + choices: List[int] # choice of quantities + + def __init__(self, quantity_details: str): + ''' + quantity_details: eg '2/[0, 2, 4, 6, 8, 12, 16]' + ''' + parts = quantity_details.split('/') + self.num_selected = int(parts[0]) + self.choices = json.loads(parts[1]) + + +class HpeV2ConfiguratorParser(IHtmlConfiguratorParser): + #
Description | + #Part Number | + #CO2e estimé | + #Price | + #Qty | + # ... + #|||||
---|---|---|---|---|---|---|---|---|---|
+ # ... + # | + + #
+ #
+ # Intel Xeon-Gold 5415+ 2.9GHz 8-core 150W Processor for HPE
+ #
+
+ #
+ #
+ # + # |
+
+ # + # P49597-B21 + # | + + #+ # | + + #
+ #
+ #
+ # + # |
+
+ #
+ #
+ #
+ #
+ #
+ # 4D
+ #
+ # |
+
+ # 09/30/2025 | + + #
+ # 133 kg CO2e
+ # |
+
+ # + # EUR 1,092.65 + # | + + #+ # + + # | + #