circuit-json-to-spice 0.0.22 → 0.0.24

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -210,6 +210,10 @@ function circuitJsonToSpice(circuitJson) {
210
210
  const netlist = new SpiceNetlist("* Circuit JSON to SPICE Netlist");
211
211
  const sourceComponents = su(circuitJson).source_component.list();
212
212
  const sourcePorts = su(circuitJson).source_port.list();
213
+ const sourceTraces = su(circuitJson).source_trace.list();
214
+ const simulationProbes = circuitJson.filter(
215
+ (elm) => elm.type === "simulation_voltage_probe"
216
+ );
213
217
  const simulationSwitches = circuitJson.filter(
214
218
  (element) => element.type === "simulation_switch"
215
219
  ).map((element) => element);
@@ -223,6 +227,18 @@ function circuitJsonToSpice(circuitJson) {
223
227
  const nodeMap = /* @__PURE__ */ new Map();
224
228
  const netToNodeName = /* @__PURE__ */ new Map();
225
229
  let nodeCounter = 1;
230
+ const probeNames = /* @__PURE__ */ new Set();
231
+ if (simulationProbes.length > 0) {
232
+ for (const probe of simulationProbes) {
233
+ if (probe.name) {
234
+ probeNames.add(probe.name);
235
+ }
236
+ }
237
+ }
238
+ const numericProbeNames = [...probeNames].map((name) => /^N(\d+)$/i.exec(name)).filter((m) => m !== null).map((m) => parseInt(m[1], 10));
239
+ if (numericProbeNames.length > 0) {
240
+ nodeCounter = Math.max(...numericProbeNames) + 1;
241
+ }
226
242
  const groundNets = /* @__PURE__ */ new Set();
227
243
  const gndSourceNetIds = new Set(
228
244
  su(circuitJson).source_net.list().filter((sn) => sn.name?.toLowerCase().includes("gnd")).map((sn) => sn.source_net_id)
@@ -262,6 +278,30 @@ function circuitJsonToSpice(circuitJson) {
262
278
  for (const groundNet of groundNets) {
263
279
  netToNodeName.set(groundNet, "0");
264
280
  }
281
+ if (simulationProbes.length > 0) {
282
+ for (const probe of simulationProbes) {
283
+ if (!probe.name) continue;
284
+ let net;
285
+ if (probe.source_port_id) {
286
+ net = connMap.getNetConnectedToId(probe.source_port_id);
287
+ } else if (probe.source_net_id) {
288
+ const trace = sourceTraces.find(
289
+ (t) => t.connected_source_net_ids.includes(probe.source_net_id)
290
+ );
291
+ if (trace && trace.connected_source_port_ids.length > 0) {
292
+ const portId = trace.connected_source_port_ids[0];
293
+ net = connMap.getNetConnectedToId(portId);
294
+ }
295
+ }
296
+ if (net) {
297
+ if (!netToNodeName.has(net)) {
298
+ netToNodeName.set(net, probe.name);
299
+ }
300
+ } else if (probe.source_port_id && probe.name) {
301
+ nodeMap.set(probe.source_port_id, probe.name);
302
+ }
303
+ }
304
+ }
265
305
  for (const port of sourcePorts) {
266
306
  const portId = port.source_port_id;
267
307
  const net = connMap.getNetConnectedToId(portId);
@@ -527,11 +567,7 @@ function circuitJsonToSpice(circuitJson) {
527
567
  (elm) => elm.type === "simulation_experiment"
528
568
  );
529
569
  if (simExperiment) {
530
- const simulationProbes = circuitJson.filter(
531
- (elm) => elm.type === "simulation_voltage_probe"
532
- );
533
570
  if (simulationProbes.length > 0) {
534
- const sourceTraces = su(circuitJson).source_trace.list();
535
571
  const nodesToProbe = /* @__PURE__ */ new Set();
536
572
  for (const probe of simulationProbes) {
537
573
  let nodeName;
@@ -547,7 +583,7 @@ function circuitJsonToSpice(circuitJson) {
547
583
  }
548
584
  }
549
585
  if (nodeName && nodeName !== "0") {
550
- nodesToProbe.add(`V(${nodeName.toLowerCase()})`);
586
+ nodesToProbe.add(`V(${nodeName})`);
551
587
  }
552
588
  }
553
589
  if (nodesToProbe.size > 0 && simExperiment.experiment_type?.includes("transient")) {
@@ -842,4 +878,4 @@ export {
842
878
  circuitJsonToSpice,
843
879
  convertSpiceNetlistToString
844
880
  };
845
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../lib/spice-utils/convertSpiceNetlistToString.ts", "../lib/spice-classes/SpiceNetlist.ts", "../lib/spice-classes/SpiceComponent.ts", "../lib/spice-commands/ResistorCommand.ts", "../lib/spice-commands/CapacitorCommand.ts", "../lib/spice-commands/VoltageSourceCommand.ts", "../lib/spice-commands/DiodeCommand.ts", "../lib/spice-commands/InductorCommand.ts", "../lib/spice-commands/VoltageControlledSwitchCommand.ts", "../lib/circuitJsonToSpice.ts", "../lib/spice-classes/SpiceSubcircuit.ts", "../lib/spice-commands/BJTCommand.ts", "../lib/spice-commands/CurrentSourceCommand.ts", "../lib/spice-commands/InductorCouplingCommand.ts", "../lib/spice-commands/JFETCommand.ts", "../lib/spice-commands/MOSFETCommand.ts", "../lib/spice-commands/SubcircuitCallCommand.ts", "../lib/spice-commands/TransmissionLineCommand.ts"],
  "sourcesContent": ["import type { SpiceNetlist } from \"../spice-classes/SpiceNetlist\"\n\nexport const convertSpiceNetlistToString = (netlist: SpiceNetlist): string => {\n  const lines: string[] = []\n\n  // Title line (required first line in SPICE)\n  lines.push(netlist.title)\n\n  // Add models\n  if (netlist.models.size > 0) {\n    lines.push(...Array.from(netlist.models.values()))\n  }\n\n  // Component lines\n  for (const component of netlist.components) {\n    lines.push(component.toSpiceString())\n  }\n\n  // Add subcircuit definitions\n  for (const subcircuit of netlist.subcircuits) {\n    lines.push(subcircuit.toSpiceString())\n  }\n\n  if (netlist.printStatements.length > 0) {\n    lines.push(...netlist.printStatements)\n  }\n\n  // Add control block if present\n  if (netlist.controls.length > 0) {\n    lines.push(\".control\")\n    lines.push(...netlist.controls)\n    lines.push(\".endc\")\n  }\n\n  if (\n    netlist.tranCommand &&\n    !lines.some((l) => l.trim().toLowerCase().startsWith(\".tran\"))\n  ) {\n    lines.push(netlist.tranCommand)\n  }\n\n  // End with .END\n  lines.push(\".END\")\n\n  return lines.join(\"\\n\")\n}\n", "import { convertSpiceNetlistToString } from \"../spice-utils/convertSpiceNetlistToString\"\nimport type { SpiceComponent } from \"./SpiceComponent\"\nimport type { SpiceSubcircuit } from \"./SpiceSubcircuit\"\n\nexport class SpiceNetlist {\n  title: string\n  components: SpiceComponent[]\n  nodes: Set<string>\n  controls: string[]\n  subcircuits: SpiceSubcircuit[]\n  models: Map<string, string>\n  tranCommand: string | null\n  printStatements: string[]\n\n  constructor(title = \"Circuit Netlist\") {\n    this.title = title\n    this.components = []\n    this.nodes = new Set()\n    this.controls = []\n    this.subcircuits = []\n    this.models = new Map()\n    this.tranCommand = null\n    this.printStatements = []\n  }\n\n  addComponent(component: SpiceComponent) {\n    this.components.push(component)\n    // Add nodes to the set\n    for (const node of component.nodes) {\n      this.nodes.add(node)\n    }\n  }\n\n  addSubcircuit(subcircuit: SpiceSubcircuit) {\n    if (this.subcircuits.find((s) => s.name === subcircuit.name)) return\n    this.subcircuits.push(subcircuit)\n  }\n\n  toSpiceString() {\n    return convertSpiceNetlistToString(this)\n  }\n}\n", "import type { BaseSpiceCommand } from \"../spice-commands/BaseSpiceCommand\"\n\nexport class SpiceComponent {\n  name: string\n  command: BaseSpiceCommand\n  nodes: string[]\n\n  constructor(name: string, command: BaseSpiceCommand, nodes: string[]) {\n    this.name = name\n    this.command = command\n    this.nodes = nodes\n  }\n\n  toSpiceString(): string {\n    return this.command.toSpiceString()\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface ResistorCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  model?: string\n  value: string\n}\n\nexport class ResistorCommand implements BaseSpiceCommand {\n  commandName = \"resistor\" as const\n  props: ResistorCommandProps\n\n  constructor(props: ResistorCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, model, value } = this.props\n    let spiceString = `R${name} ${positiveNode} ${negativeNode}`\n    if (model) {\n      spiceString += ` ${model}`\n    }\n    spiceString += ` ${value}`\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface CapacitorCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  modelName?: string\n  value: string\n  initialCondition?: string\n}\n\nexport class CapacitorCommand implements BaseSpiceCommand {\n  commandName = \"capacitor\" as const\n\n  props: CapacitorCommandProps\n\n  constructor(props: CapacitorCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const {\n      name,\n      positiveNode,\n      negativeNode,\n      modelName,\n      value,\n      initialCondition,\n    } = this.props\n\n    let spiceString = `C${name} ${positiveNode} ${negativeNode}`\n    if (modelName) {\n      spiceString += ` ${modelName}`\n    }\n    spiceString += ` ${value}`\n    if (initialCondition) {\n      spiceString += ` IC=${initialCondition}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface VoltageSourceCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  value?: string\n  acMagnitude?: string\n  acPhase?: string\n}\n\nexport class VoltageSourceCommand implements BaseSpiceCommand {\n  commandName = \"voltage_source\" as const\n  props: VoltageSourceCommandProps\n\n  constructor(props: VoltageSourceCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, value, acMagnitude, acPhase } =\n      this.props\n    let spiceString = `V${name} ${positiveNode} ${negativeNode}`\n    if (value) {\n      spiceString += ` ${value}`\n    }\n    if (acMagnitude) {\n      spiceString += ` AC ${acMagnitude}`\n      if (acPhase) {\n        spiceString += ` ${acPhase}`\n      }\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface DiodeCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  model: string\n  area?: string\n}\n\nexport class DiodeCommand implements BaseSpiceCommand {\n  commandName = \"diode\" as const\n  props: DiodeCommandProps\n\n  constructor(props: DiodeCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, model, area } = this.props\n    let spiceString = `D${name} ${positiveNode} ${negativeNode} ${model}`\n    if (area) {\n      spiceString += ` ${area}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface InductorCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  model?: string\n  value: string\n  initialCondition?: string\n}\n\nexport class InductorCommand implements BaseSpiceCommand {\n  commandName = \"inductor\" as const\n  props: InductorCommandProps\n\n  constructor(props: InductorCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, model, value, initialCondition } =\n      this.props\n    let spiceString = `L${name} ${positiveNode} ${negativeNode}`\n    if (model) {\n      spiceString += ` ${model}`\n    }\n    spiceString += ` ${value}`\n    if (initialCondition) {\n      spiceString += ` IC=${initialCondition}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface VoltageControlledSwitchCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  positiveControl: string\n  negativeControl: string\n  model: string\n}\n\nexport class VoltageControlledSwitchCommand implements BaseSpiceCommand {\n  commandName = \"voltage_controlled_switch\" as const\n  props: VoltageControlledSwitchCommandProps\n\n  constructor(props: VoltageControlledSwitchCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const {\n      name,\n      positiveNode,\n      negativeNode,\n      positiveControl,\n      negativeControl,\n      model,\n    } = this.props\n    return `S${name} ${positiveNode} ${negativeNode} ${positiveControl} ${negativeControl} ${model}`\n  }\n}\n", "import { SpiceNetlist } from \"./spice-classes/SpiceNetlist\"\nimport { SpiceComponent } from \"./spice-classes/SpiceComponent\"\nimport { ResistorCommand } from \"./spice-commands/ResistorCommand\"\nimport { CapacitorCommand } from \"./spice-commands/CapacitorCommand\"\nimport { VoltageSourceCommand } from \"./spice-commands/VoltageSourceCommand\"\nimport { DiodeCommand } from \"./spice-commands/DiodeCommand\"\nimport { InductorCommand } from \"./spice-commands/InductorCommand\"\nimport { VoltageControlledSwitchCommand } from \"./spice-commands/VoltageControlledSwitchCommand\"\nimport type { AnyCircuitElement, SimulationSwitch } from \"circuit-json\"\nimport { getSourcePortConnectivityMapFromCircuitJson } from \"circuit-json-to-connectivity-map\"\nimport { su } from \"@tscircuit/circuit-json-util\"\n\nexport function circuitJsonToSpice(\n  circuitJson: AnyCircuitElement[],\n): SpiceNetlist {\n  const netlist = new SpiceNetlist(\"* Circuit JSON to SPICE Netlist\")\n  const sourceComponents = su(circuitJson).source_component.list()\n  const sourcePorts = su(circuitJson).source_port.list()\n  const simulationSwitches = circuitJson\n    .filter(\n      (element) => (element as { type?: string }).type === \"simulation_switch\",\n    )\n    .map((element) => element as unknown as SimulationSwitch)\n  const simulationSwitchMap = new Map<string, SimulationSwitch>()\n\n  for (const simSwitch of simulationSwitches) {\n    if (simSwitch.source_component_id) {\n      simulationSwitchMap.set(simSwitch.source_component_id, simSwitch)\n    }\n  }\n\n  const connMap = getSourcePortConnectivityMapFromCircuitJson(circuitJson)\n\n  // Create node mapping from port connections\n  const nodeMap = new Map<string, string>()\n  const netToNodeName = new Map<string, string>()\n  let nodeCounter = 1\n\n  const groundNets = new Set<string>()\n\n  // Find ground from source nets that include \"gnd\" in the name\n  const gndSourceNetIds = new Set(\n    su(circuitJson)\n      .source_net.list()\n      .filter((sn) => sn.name?.toLowerCase().includes(\"gnd\"))\n      .map((sn) => sn.source_net_id),\n  )\n\n  if (gndSourceNetIds.size > 0) {\n    for (const trace of su(circuitJson).source_trace.list()) {\n      if (trace.connected_source_port_ids.length > 0) {\n        const isOnGndNet = trace.connected_source_net_ids.some((netId) =>\n          gndSourceNetIds.has(netId),\n        )\n        if (isOnGndNet) {\n          const aPortOnGnd = trace.connected_source_port_ids[0]\n          const gndNet = connMap.getNetConnectedToId(aPortOnGnd)\n          if (gndNet) {\n            groundNets.add(gndNet)\n          }\n        }\n      }\n    }\n  }\n\n  // Find ground node from ports named \"GND\"\n  const groundPorts = sourcePorts.filter((p) => p.name?.toLowerCase() === \"gnd\")\n  for (const groundPort of groundPorts) {\n    const groundNet = connMap.getNetConnectedToId(groundPort.source_port_id)\n    if (groundNet) {\n      groundNets.add(groundNet)\n    }\n  }\n\n  for (const simSource of su(circuitJson).simulation_voltage_source.list()) {\n    const neg_port_id =\n      (simSource as any).negative_source_port_id ??\n      (simSource as any).terminal2_source_port_id\n    if (neg_port_id) {\n      const gnd_net = connMap.getNetConnectedToId(neg_port_id)\n      if (gnd_net) {\n        groundNets.add(gnd_net)\n      }\n    }\n  }\n\n  for (const groundNet of groundNets) {\n    netToNodeName.set(groundNet, \"0\")\n  }\n\n  // First pass: assign node numbers to all connected nets\n  for (const port of sourcePorts) {\n    const portId = port.source_port_id\n    const net = connMap.getNetConnectedToId(portId)\n    if (net) {\n      if (!netToNodeName.has(net)) {\n        netToNodeName.set(net, `N${nodeCounter++}`)\n      }\n      nodeMap.set(portId, netToNodeName.get(net)!)\n    }\n  }\n\n  // Second pass: assign node numbers to unconnected ports\n  for (const port of sourcePorts) {\n    const portId = port.source_port_id\n    // If a port wasn't in a net, it won't be in the nodeMap yet\n    if (!nodeMap.has(portId)) {\n      // Unconnected port, create a new floating node for it\n      nodeMap.set(portId, `N${nodeCounter++}`)\n    }\n  }\n\n  // Process each component\n  for (const component of sourceComponents) {\n    if (component.type !== \"source_component\") continue\n\n    const componentPorts = su(circuitJson)\n      .source_port.list({\n        source_component_id: component.source_component_id,\n      })\n      .sort((a, b) => (a.pin_number ?? 0) - (b.pin_number ?? 0))\n\n    // Get node names for component ports\n    const nodes = componentPorts.map((port) => {\n      return nodeMap.get(port.source_port_id) || \"0\"\n    })\n\n    // Create SPICE component based on type\n    if (\"ftype\" in component) {\n      let spiceComponent: SpiceComponent | null = null\n\n      switch (component.ftype) {\n        case \"simple_resistor\": {\n          if (\"resistance\" in component && \"name\" in component) {\n            const resistorCmd = new ResistorCommand({\n              name: component.name,\n              positiveNode: nodes[0] || \"0\",\n              negativeNode: nodes[1] || \"0\",\n              value: formatResistance(component.resistance),\n            })\n            spiceComponent = new SpiceComponent(\n              component.name,\n              resistorCmd,\n              nodes,\n            )\n          }\n          break\n        }\n        case \"simple_switch\": {\n          const sanitizedBase = sanitizeIdentifier(\n            component.name ?? component.source_component_id,\n            \"SW\",\n          )\n          const positiveNode = nodes[0] || \"0\"\n          const negativeNode = nodes[1] || \"0\"\n          const controlNode = `NCTRL_${sanitizedBase}`\n          const modelName = `SW_${sanitizedBase}`\n\n          const associatedSimulationSwitch = simulationSwitchMap.get(\n            component.source_component_id,\n          )\n\n          const controlValue = buildSimulationSwitchControlValue(\n            associatedSimulationSwitch,\n          )\n\n          const switchCmd = new VoltageControlledSwitchCommand({\n            name: sanitizedBase,\n            positiveNode,\n            negativeNode,\n            positiveControl: controlNode,\n            negativeControl: \"0\",\n            model: modelName,\n          })\n\n          spiceComponent = new SpiceComponent(sanitizedBase, switchCmd, [\n            positiveNode,\n            negativeNode,\n            controlNode,\n            \"0\",\n          ])\n\n          if (!netlist.models.has(modelName)) {\n            netlist.models.set(\n              modelName,\n              `.MODEL ${modelName} SW(Ron=0.1 Roff=1e9 Vt=2.5 Vh=0.1)`,\n            )\n          }\n\n          const controlSourceName = `CTRL_${sanitizedBase}`\n          const controlSourceCmd = new VoltageSourceCommand({\n            name: controlSourceName,\n            positiveNode: controlNode,\n            negativeNode: \"0\",\n            value: controlValue,\n          })\n\n          const controlComponent = new SpiceComponent(\n            controlSourceName,\n            controlSourceCmd,\n            [controlNode, \"0\"],\n          )\n\n          netlist.addComponent(controlComponent)\n          break\n        }\n\n        case \"simple_capacitor\": {\n          if (\"capacitance\" in component && \"name\" in component) {\n            const capacitorCmd = new CapacitorCommand({\n              name: component.name,\n              positiveNode: nodes[0] || \"0\",\n              negativeNode: nodes[1] || \"0\",\n              value: formatCapacitance(component.capacitance),\n            })\n            spiceComponent = new SpiceComponent(\n              component.name,\n              capacitorCmd,\n              nodes,\n            )\n          }\n          break\n        }\n        case \"simple_diode\": {\n          if (\"name\" in component) {\n            const anodePort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"anode\" ||\n                p.port_hints?.includes(\"anode\"),\n            )\n            const cathodePort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"cathode\" ||\n                p.port_hints?.includes(\"cathode\"),\n            )\n            const positiveNode =\n              nodeMap.get(anodePort?.source_port_id ?? \"\") || \"0\"\n            const negativeNode =\n              nodeMap.get(cathodePort?.source_port_id ?? \"\") || \"0\"\n\n            const modelName = \"D\"\n            const diodeCmd = new DiodeCommand({\n              name: component.name,\n              positiveNode,\n              negativeNode,\n              model: modelName, // generic model\n            })\n            netlist.models.set(modelName, `.MODEL ${modelName} D`)\n            spiceComponent = new SpiceComponent(component.name, diodeCmd, [\n              positiveNode,\n              negativeNode,\n            ])\n          }\n          break\n        }\n        case \"simple_inductor\": {\n          if (\"inductance\" in component && \"name\" in component) {\n            const inductorCmd = new InductorCommand({\n              name: component.name,\n              positiveNode: nodes[0] || \"0\",\n              negativeNode: nodes[1] || \"0\",\n              value: formatInductance(component.inductance),\n            })\n            spiceComponent = new SpiceComponent(\n              component.name,\n              inductorCmd,\n              nodes,\n            )\n          }\n          break\n        }\n        case \"simple_mosfet\": {\n          if (\"name\" in component) {\n            const drainPort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"drain\" ||\n                p.port_hints?.includes(\"drain\"),\n            )\n            const gatePort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"gate\" ||\n                p.port_hints?.includes(\"gate\"),\n            )\n            const sourcePort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"source\" ||\n                p.port_hints?.includes(\"source\"),\n            )\n\n            const drainNode =\n              nodeMap.get(drainPort?.source_port_id ?? \"\") || \"0\"\n            const gateNode = nodeMap.get(gatePort?.source_port_id ?? \"\") || \"0\"\n            const sourceNode =\n              nodeMap.get(sourcePort?.source_port_id ?? \"\") || \"0\"\n\n            const modelName = \"SWMOD\"\n            const switchCmd = new VoltageControlledSwitchCommand({\n              name: component.name,\n              positiveNode: drainNode,\n              negativeNode: sourceNode,\n              positiveControl: gateNode,\n              negativeControl: sourceNode,\n              model: modelName,\n            })\n            netlist.models.set(modelName, `.MODEL ${modelName} SW`)\n\n            spiceComponent = new SpiceComponent(component.name, switchCmd, [\n              drainNode,\n              gateNode,\n              sourceNode,\n            ])\n          }\n          break\n        }\n      }\n\n      if (spiceComponent) {\n        netlist.addComponent(spiceComponent)\n      }\n    }\n  }\n\n  // Process simulation voltage sources\n  const simulationVoltageSources =\n    su(circuitJson).simulation_voltage_source.list()\n\n  for (const simSource of simulationVoltageSources) {\n    if (simSource.type !== \"simulation_voltage_source\") continue\n\n    if ((simSource as any).is_dc_source === false) {\n      // AC Source\n      if (\n        \"terminal1_source_port_id\" in simSource &&\n        \"terminal2_source_port_id\" in simSource &&\n        (simSource as any).terminal1_source_port_id &&\n        (simSource as any).terminal2_source_port_id\n      ) {\n        const positiveNode =\n          nodeMap.get((simSource as any).terminal1_source_port_id) || \"0\"\n        const negativeNode =\n          nodeMap.get((simSource as any).terminal2_source_port_id) || \"0\"\n\n        let value = \"\"\n        const wave_shape = (simSource as any).wave_shape\n        if (wave_shape === \"sinewave\") {\n          const v_offset = 0 // not provided in circuitJson\n          const v_peak = (simSource as any).voltage ?? 0\n          const freq = (simSource as any).frequency ?? 0\n          const delay = 0 // not provided in circuitJson\n          const damping_factor = 0 // not provided in circuitJson\n          const phase = (simSource as any).phase ?? 0\n          if (freq > 0) {\n            value = `SIN(${v_offset} ${v_peak} ${freq} ${delay} ${damping_factor} ${phase})`\n          } else {\n            value = `DC ${(simSource as any).voltage ?? 0}`\n          }\n        } else if (wave_shape === \"square\") {\n          const v_initial = 0\n          const v_pulsed = (simSource as any).voltage ?? 0\n          const freq = (simSource as any).frequency ?? 0\n          const period_from_freq = freq === 0 ? Infinity : 1 / freq\n          const period = (simSource as any).period ?? period_from_freq\n          const duty_cycle = (simSource as any).duty_cycle ?? 0.5\n          const pulse_width = period * duty_cycle\n          const delay = 0\n          const rise_time = \"1n\"\n          const fall_time = \"1n\"\n          value = `PULSE(${v_initial} ${v_pulsed} ${delay} ${rise_time} ${fall_time} ${pulse_width} ${period})`\n        } else if ((simSource as any).voltage !== undefined) {\n          value = `DC ${(simSource as any).voltage}`\n        }\n\n        if (value) {\n          const voltageSourceCmd = new VoltageSourceCommand({\n            name: simSource.simulation_voltage_source_id,\n            positiveNode,\n            negativeNode,\n            value,\n          })\n\n          const spiceComponent = new SpiceComponent(\n            simSource.simulation_voltage_source_id,\n            voltageSourceCmd,\n            [positiveNode, negativeNode],\n          )\n          netlist.addComponent(spiceComponent)\n        }\n      }\n    } else {\n      // DC Source (is_dc_source is true or undefined)\n      const positivePortId =\n        (simSource as any).positive_source_port_id ??\n        (simSource as any).terminal1_source_port_id\n      const negativePortId =\n        (simSource as any).negative_source_port_id ??\n        (simSource as any).terminal2_source_port_id\n\n      if (\n        positivePortId &&\n        negativePortId &&\n        \"voltage\" in simSource &&\n        (simSource as any).voltage !== undefined\n      ) {\n        const positiveNode = nodeMap.get(positivePortId) || \"0\"\n        const negativeNode = nodeMap.get(negativePortId) || \"0\"\n\n        const voltageSourceCmd = new VoltageSourceCommand({\n          name: simSource.simulation_voltage_source_id,\n          positiveNode,\n          negativeNode,\n          value: `DC ${(simSource as any).voltage}`,\n        })\n\n        const spiceComponent = new SpiceComponent(\n          simSource.simulation_voltage_source_id,\n          voltageSourceCmd,\n          [positiveNode, negativeNode],\n        )\n        netlist.addComponent(spiceComponent)\n      }\n    }\n  }\n\n  const simExperiment = circuitJson.find(\n    (elm) => elm.type === \"simulation_experiment\",\n  )\n\n  if (simExperiment) {\n    // Process simulation voltage probes\n    const simulationProbes = circuitJson.filter(\n      (elm) => elm.type === \"simulation_voltage_probe\",\n    ) as unknown as Array<{\n      type: \"simulation_voltage_probe\"\n      source_port_id?: string\n      source_net_id?: string\n    }>\n\n    if (simulationProbes.length > 0) {\n      const sourceTraces = su(circuitJson).source_trace.list()\n      const nodesToProbe = new Set<string>()\n\n      for (const probe of simulationProbes) {\n        let nodeName: string | undefined\n        if (probe.source_port_id) {\n          nodeName = nodeMap.get(probe.source_port_id)\n        } else if (probe.source_net_id) {\n          const trace = sourceTraces.find((t) =>\n            t.connected_source_net_ids.includes(probe.source_net_id!),\n          )\n          if (trace && trace.connected_source_port_ids.length > 0) {\n            const portId = trace.connected_source_port_ids[0]\n            nodeName = nodeMap.get(portId)\n          }\n        }\n\n        if (nodeName && nodeName !== \"0\") {\n          nodesToProbe.add(`V(${nodeName.toLowerCase()})`)\n        }\n      }\n\n      if (\n        nodesToProbe.size > 0 &&\n        (simExperiment as any).experiment_type?.includes(\"transient\")\n      ) {\n        netlist.printStatements.push(\n          `.PRINT TRAN ${[...nodesToProbe].join(\" \")}`,\n        )\n      }\n    }\n\n    const timePerStep = (simExperiment as any).time_per_step\n    const endTime = (simExperiment as any).end_time_ms\n    const startTimeMs = (simExperiment as any).start_time_ms\n\n    if (timePerStep && endTime) {\n      // circuit-json values are in ms, SPICE requires seconds\n      const startTime = (startTimeMs ?? 0) / 1000\n\n      let tranCmd = `.tran ${formatNumberForSpice(\n        timePerStep / 1000,\n      )} ${formatNumberForSpice(endTime / 1000)}`\n      if (startTime > 0) {\n        tranCmd += ` ${formatNumberForSpice(startTime)}`\n      }\n      tranCmd += \" UIC\"\n      netlist.tranCommand = tranCmd\n    }\n  }\n\n  return netlist\n}\n\nfunction formatResistance(resistance: number): string {\n  if (resistance >= 1e6) return `${resistance / 1e6}MEG`\n  if (resistance >= 1e3) return `${resistance / 1e3}K`\n  return resistance.toString()\n}\n\nfunction formatCapacitance(capacitance: number): string {\n  if (capacitance >= 1e-3) return `${capacitance * 1e3}M`\n  if (capacitance >= 1e-6) return `${capacitance * 1e6}U`\n  if (capacitance >= 1e-9) return `${capacitance * 1e9}N`\n  if (capacitance >= 1e-12) return `${capacitance * 1e12}P`\n  return capacitance.toString()\n}\n\nfunction formatInductance(inductance: number): string {\n  if (inductance >= 1) return inductance.toString()\n  if (inductance >= 1e-3) return `${inductance * 1e3}m`\n  if (inductance >= 1e-6) return `${inductance * 1e6}u`\n  if (inductance >= 1e-9) return `${inductance * 1e9}n`\n  if (inductance >= 1e-12) return `${inductance * 1e12}p`\n  return inductance.toString()\n}\n\nfunction sanitizeIdentifier(value: string | undefined, prefix: string) {\n  if (!value) return prefix\n  const sanitized = value.replace(/[^A-Za-z0-9_]/g, \"_\")\n  if (!sanitized) return prefix\n  if (/^[0-9]/.test(sanitized)) {\n    return `${prefix}_${sanitized}`\n  }\n  return sanitized\n}\n\nfunction buildSimulationSwitchControlValue(\n  simulationSwitch: SimulationSwitch | undefined,\n) {\n  const highVoltage = 5\n  const lowVoltage = 0\n  const riseTime = \"1n\"\n  const fallTime = \"1n\"\n\n  if (!simulationSwitch) {\n    return `DC ${lowVoltage}`\n  }\n\n  const startsClosed = simulationSwitch.starts_closed ?? false\n  const closesAt = simulationSwitch.closes_at ?? 0\n  const opensAt = simulationSwitch.opens_at\n  const switchingFrequency = simulationSwitch.switching_frequency\n\n  const [initialVoltage, pulsedVoltage] = startsClosed\n    ? [highVoltage, lowVoltage]\n    : [lowVoltage, highVoltage]\n\n  if (switchingFrequency && switchingFrequency > 0) {\n    const period = 1 / switchingFrequency\n    const widthFromOpenClose =\n      opensAt && opensAt > closesAt ? Math.min(opensAt - closesAt, period) : 0\n    const pulseWidth =\n      widthFromOpenClose > 0 ? widthFromOpenClose : Math.max(period / 2, 1e-9)\n\n    return `PULSE(${formatNumberForSpice(initialVoltage)} ${formatNumberForSpice(pulsedVoltage)} ${formatNumberForSpice(closesAt)} ${riseTime} ${fallTime} ${formatNumberForSpice(pulseWidth)} ${formatNumberForSpice(period)})`\n  }\n\n  if (opensAt !== undefined && opensAt > closesAt) {\n    const pulseWidth = Math.max(opensAt - closesAt, 1e-9)\n    const period = closesAt + pulseWidth * 2\n\n    return `PULSE(${formatNumberForSpice(initialVoltage)} ${formatNumberForSpice(pulsedVoltage)} ${formatNumberForSpice(closesAt)} ${riseTime} ${fallTime} ${formatNumberForSpice(pulseWidth)} ${formatNumberForSpice(period)})`\n  }\n\n  if (closesAt > 0) {\n    const period = closesAt * 2\n    const pulseWidth = Math.max(period / 2, 1e-9)\n    return `PULSE(${formatNumberForSpice(initialVoltage)} ${formatNumberForSpice(pulsedVoltage)} ${formatNumberForSpice(closesAt)} ${riseTime} ${fallTime} ${formatNumberForSpice(pulseWidth)} ${formatNumberForSpice(period)})`\n  }\n\n  return `DC ${startsClosed ? highVoltage : lowVoltage}`\n}\n\nfunction formatNumberForSpice(value: number) {\n  if (!Number.isFinite(value)) return `${value}`\n  if (value === 0) return \"0\"\n\n  const absValue = Math.abs(value)\n\n  if (absValue >= 1e3 || absValue <= 1e-3) {\n    return Number(value.toExponential(6)).toString()\n  }\n\n  return Number(value.toPrecision(6)).toString()\n}\n", "export class SpiceSubcircuit {\n  name: string\n  pins: string[]\n\n  constructor(name: string, pins: string[]) {\n    this.name = name\n    this.pins = pins\n  }\n\n  toSpiceString(): string {\n    const pinString = this.pins.join(\" \")\n    const header = `.SUBCKT ${this.name} ${pinString}`\n    const footer = `.ENDS ${this.name}`\n\n    const body = `* Placeholder for ${this.name}. No definition found in circuit JSON.`\n\n    return [\"\", header, body, footer].join(\"\\n\")\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface BJTCommandProps {\n  name: string\n  collector: string\n  base: string\n  emitter: string\n  substrate?: string\n  model: string\n  area?: string\n}\n\nexport class BJTCommand implements BaseSpiceCommand {\n  commandName = \"bjt\" as const\n  props: BJTCommandProps\n\n  constructor(props: BJTCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, collector, base, emitter, substrate, model, area } =\n      this.props\n    let spiceString = `Q${name} ${collector} ${base} ${emitter}`\n    if (substrate) {\n      spiceString += ` ${substrate}`\n    }\n    spiceString += ` ${model}`\n    if (area) {\n      spiceString += ` ${area}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface CurrentSourceCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  value?: string\n  acMagnitude?: string\n  acPhase?: string\n}\n\nexport class CurrentSourceCommand implements BaseSpiceCommand {\n  commandName = \"current_source\" as const\n  props: CurrentSourceCommandProps\n\n  constructor(props: CurrentSourceCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, value, acMagnitude, acPhase } =\n      this.props\n    let spiceString = `I${name} ${positiveNode} ${negativeNode}`\n    if (value) {\n      spiceString += ` ${value}`\n    }\n    if (acMagnitude) {\n      spiceString += ` AC ${acMagnitude}`\n      if (acPhase) {\n        spiceString += ` ${acPhase}`\n      }\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface InductorCouplingCommandProps {\n  name: string\n  inductors: string[]\n  coupling: string\n}\n\nexport class InductorCouplingCommand implements BaseSpiceCommand {\n  commandName = \"inductor_coupling\" as const\n  props: InductorCouplingCommandProps\n\n  constructor(props: InductorCouplingCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, inductors, coupling } = this.props\n    return `K${name} ${inductors.join(\" \")} ${coupling}`\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface JFETCommandProps {\n  name: string\n  drain: string\n  gate: string\n  source: string\n  model: string\n  area?: string\n}\n\nexport class JFETCommand implements BaseSpiceCommand {\n  commandName = \"jfet\" as const\n  props: JFETCommandProps\n\n  constructor(props: JFETCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, drain, gate, source, model, area } = this.props\n    let spiceString = `J${name} ${drain} ${gate} ${source} ${model}`\n    if (area) {\n      spiceString += ` ${area}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface MOSFETCommandProps {\n  name: string\n  drain: string\n  gate: string\n  source: string\n  substrate: string\n  model: string\n  length?: string\n  width?: string\n  drainArea?: string\n  sourceArea?: string\n  drainPerimeter?: string\n  sourcePerimeter?: string\n  drainResistance?: string\n  sourceResistance?: string\n}\n\nexport class MOSFETCommand implements BaseSpiceCommand {\n  commandName = \"mosfet\" as const\n  props: MOSFETCommandProps\n\n  constructor(props: MOSFETCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const {\n      name,\n      drain,\n      gate,\n      source,\n      substrate,\n      model,\n      length,\n      width,\n      drainArea,\n      sourceArea,\n      drainPerimeter,\n      sourcePerimeter,\n      drainResistance,\n      sourceResistance,\n    } = this.props\n\n    let spiceString = `M${name} ${drain} ${gate} ${source} ${substrate} ${model}`\n\n    const params: Record<string, string | undefined> = {\n      L: length,\n      W: width,\n      AD: drainArea,\n      AS: sourceArea,\n      PD: drainPerimeter,\n      PS: sourcePerimeter,\n      NRD: drainResistance,\n      NRS: sourceResistance,\n    }\n\n    Object.entries(params).forEach(([key, value]) => {\n      if (value) {\n        spiceString += ` ${key}=${value}`\n      }\n    })\n\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface SubcircuitCallCommandProps {\n  name: string\n  nodes: string[]\n  subcircuitName: string\n}\n\nexport class SubcircuitCallCommand implements BaseSpiceCommand {\n  commandName = \"subcircuit_call\" as const\n  props: SubcircuitCallCommandProps\n\n  constructor(props: SubcircuitCallCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, nodes, subcircuitName } = this.props\n    return `X${name} ${nodes.join(\" \")} ${subcircuitName}`\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface TransmissionLineCommandProps {\n  name: string\n  aPositive: string\n  aNegative: string\n  bPositive: string\n  bNegative: string\n  impedance: string\n  delay?: string\n  frequency?: string\n  normalizedLength?: string\n}\n\nexport class TransmissionLineCommand implements BaseSpiceCommand {\n  commandName = \"transmission_line\" as const\n  props: TransmissionLineCommandProps\n\n  constructor(props: TransmissionLineCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const {\n      name,\n      aPositive,\n      aNegative,\n      bPositive,\n      bNegative,\n      impedance,\n      delay,\n      frequency,\n      normalizedLength,\n    } = this.props\n    let spiceString = `T${name} ${aPositive} ${aNegative} ${bPositive} ${bNegative} Z0=${impedance}`\n    if (delay) {\n      spiceString += ` TD=${delay}`\n    } else if (frequency) {\n      spiceString += ` F=${frequency}`\n      if (normalizedLength) {\n        spiceString += ` NL=${normalizedLength}`\n      }\n    }\n    return spiceString\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

881
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../lib/spice-utils/convertSpiceNetlistToString.ts", "../lib/spice-classes/SpiceNetlist.ts", "../lib/spice-classes/SpiceComponent.ts", "../lib/spice-commands/ResistorCommand.ts", "../lib/spice-commands/CapacitorCommand.ts", "../lib/spice-commands/VoltageSourceCommand.ts", "../lib/spice-commands/DiodeCommand.ts", "../lib/spice-commands/InductorCommand.ts", "../lib/spice-commands/VoltageControlledSwitchCommand.ts", "../lib/circuitJsonToSpice.ts", "../lib/spice-classes/SpiceSubcircuit.ts", "../lib/spice-commands/BJTCommand.ts", "../lib/spice-commands/CurrentSourceCommand.ts", "../lib/spice-commands/InductorCouplingCommand.ts", "../lib/spice-commands/JFETCommand.ts", "../lib/spice-commands/MOSFETCommand.ts", "../lib/spice-commands/SubcircuitCallCommand.ts", "../lib/spice-commands/TransmissionLineCommand.ts"],
  "sourcesContent": ["import type { SpiceNetlist } from \"../spice-classes/SpiceNetlist\"\n\nexport const convertSpiceNetlistToString = (netlist: SpiceNetlist): string => {\n  const lines: string[] = []\n\n  // Title line (required first line in SPICE)\n  lines.push(netlist.title)\n\n  // Add models\n  if (netlist.models.size > 0) {\n    lines.push(...Array.from(netlist.models.values()))\n  }\n\n  // Component lines\n  for (const component of netlist.components) {\n    lines.push(component.toSpiceString())\n  }\n\n  // Add subcircuit definitions\n  for (const subcircuit of netlist.subcircuits) {\n    lines.push(subcircuit.toSpiceString())\n  }\n\n  if (netlist.printStatements.length > 0) {\n    lines.push(...netlist.printStatements)\n  }\n\n  // Add control block if present\n  if (netlist.controls.length > 0) {\n    lines.push(\".control\")\n    lines.push(...netlist.controls)\n    lines.push(\".endc\")\n  }\n\n  if (\n    netlist.tranCommand &&\n    !lines.some((l) => l.trim().toLowerCase().startsWith(\".tran\"))\n  ) {\n    lines.push(netlist.tranCommand)\n  }\n\n  // End with .END\n  lines.push(\".END\")\n\n  return lines.join(\"\\n\")\n}\n", "import { convertSpiceNetlistToString } from \"../spice-utils/convertSpiceNetlistToString\"\nimport type { SpiceComponent } from \"./SpiceComponent\"\nimport type { SpiceSubcircuit } from \"./SpiceSubcircuit\"\n\nexport class SpiceNetlist {\n  title: string\n  components: SpiceComponent[]\n  nodes: Set<string>\n  controls: string[]\n  subcircuits: SpiceSubcircuit[]\n  models: Map<string, string>\n  tranCommand: string | null\n  printStatements: string[]\n\n  constructor(title = \"Circuit Netlist\") {\n    this.title = title\n    this.components = []\n    this.nodes = new Set()\n    this.controls = []\n    this.subcircuits = []\n    this.models = new Map()\n    this.tranCommand = null\n    this.printStatements = []\n  }\n\n  addComponent(component: SpiceComponent) {\n    this.components.push(component)\n    // Add nodes to the set\n    for (const node of component.nodes) {\n      this.nodes.add(node)\n    }\n  }\n\n  addSubcircuit(subcircuit: SpiceSubcircuit) {\n    if (this.subcircuits.find((s) => s.name === subcircuit.name)) return\n    this.subcircuits.push(subcircuit)\n  }\n\n  toSpiceString() {\n    return convertSpiceNetlistToString(this)\n  }\n}\n", "import type { BaseSpiceCommand } from \"../spice-commands/BaseSpiceCommand\"\n\nexport class SpiceComponent {\n  name: string\n  command: BaseSpiceCommand\n  nodes: string[]\n\n  constructor(name: string, command: BaseSpiceCommand, nodes: string[]) {\n    this.name = name\n    this.command = command\n    this.nodes = nodes\n  }\n\n  toSpiceString(): string {\n    return this.command.toSpiceString()\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface ResistorCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  model?: string\n  value: string\n}\n\nexport class ResistorCommand implements BaseSpiceCommand {\n  commandName = \"resistor\" as const\n  props: ResistorCommandProps\n\n  constructor(props: ResistorCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, model, value } = this.props\n    let spiceString = `R${name} ${positiveNode} ${negativeNode}`\n    if (model) {\n      spiceString += ` ${model}`\n    }\n    spiceString += ` ${value}`\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface CapacitorCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  modelName?: string\n  value: string\n  initialCondition?: string\n}\n\nexport class CapacitorCommand implements BaseSpiceCommand {\n  commandName = \"capacitor\" as const\n\n  props: CapacitorCommandProps\n\n  constructor(props: CapacitorCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const {\n      name,\n      positiveNode,\n      negativeNode,\n      modelName,\n      value,\n      initialCondition,\n    } = this.props\n\n    let spiceString = `C${name} ${positiveNode} ${negativeNode}`\n    if (modelName) {\n      spiceString += ` ${modelName}`\n    }\n    spiceString += ` ${value}`\n    if (initialCondition) {\n      spiceString += ` IC=${initialCondition}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface VoltageSourceCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  value?: string\n  acMagnitude?: string\n  acPhase?: string\n}\n\nexport class VoltageSourceCommand implements BaseSpiceCommand {\n  commandName = \"voltage_source\" as const\n  props: VoltageSourceCommandProps\n\n  constructor(props: VoltageSourceCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, value, acMagnitude, acPhase } =\n      this.props\n    let spiceString = `V${name} ${positiveNode} ${negativeNode}`\n    if (value) {\n      spiceString += ` ${value}`\n    }\n    if (acMagnitude) {\n      spiceString += ` AC ${acMagnitude}`\n      if (acPhase) {\n        spiceString += ` ${acPhase}`\n      }\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface DiodeCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  model: string\n  area?: string\n}\n\nexport class DiodeCommand implements BaseSpiceCommand {\n  commandName = \"diode\" as const\n  props: DiodeCommandProps\n\n  constructor(props: DiodeCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, model, area } = this.props\n    let spiceString = `D${name} ${positiveNode} ${negativeNode} ${model}`\n    if (area) {\n      spiceString += ` ${area}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface InductorCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  model?: string\n  value: string\n  initialCondition?: string\n}\n\nexport class InductorCommand implements BaseSpiceCommand {\n  commandName = \"inductor\" as const\n  props: InductorCommandProps\n\n  constructor(props: InductorCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, model, value, initialCondition } =\n      this.props\n    let spiceString = `L${name} ${positiveNode} ${negativeNode}`\n    if (model) {\n      spiceString += ` ${model}`\n    }\n    spiceString += ` ${value}`\n    if (initialCondition) {\n      spiceString += ` IC=${initialCondition}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface VoltageControlledSwitchCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  positiveControl: string\n  negativeControl: string\n  model: string\n}\n\nexport class VoltageControlledSwitchCommand implements BaseSpiceCommand {\n  commandName = \"voltage_controlled_switch\" as const\n  props: VoltageControlledSwitchCommandProps\n\n  constructor(props: VoltageControlledSwitchCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const {\n      name,\n      positiveNode,\n      negativeNode,\n      positiveControl,\n      negativeControl,\n      model,\n    } = this.props\n    return `S${name} ${positiveNode} ${negativeNode} ${positiveControl} ${negativeControl} ${model}`\n  }\n}\n", "import { SpiceNetlist } from \"./spice-classes/SpiceNetlist\"\nimport { SpiceComponent } from \"./spice-classes/SpiceComponent\"\nimport { ResistorCommand } from \"./spice-commands/ResistorCommand\"\nimport { CapacitorCommand } from \"./spice-commands/CapacitorCommand\"\nimport { VoltageSourceCommand } from \"./spice-commands/VoltageSourceCommand\"\nimport { DiodeCommand } from \"./spice-commands/DiodeCommand\"\nimport { InductorCommand } from \"./spice-commands/InductorCommand\"\nimport { VoltageControlledSwitchCommand } from \"./spice-commands/VoltageControlledSwitchCommand\"\nimport type {\n  AnyCircuitElement,\n  SimulationSwitch,\n  SimulationVoltageProbe,\n} from \"circuit-json\"\nimport { getSourcePortConnectivityMapFromCircuitJson } from \"circuit-json-to-connectivity-map\"\nimport { su } from \"@tscircuit/circuit-json-util\"\n\nexport function circuitJsonToSpice(\n  circuitJson: AnyCircuitElement[],\n): SpiceNetlist {\n  const netlist = new SpiceNetlist(\"* Circuit JSON to SPICE Netlist\")\n  const sourceComponents = su(circuitJson).source_component.list()\n  const sourcePorts = su(circuitJson).source_port.list()\n  const sourceTraces = su(circuitJson).source_trace.list()\n  const simulationProbes = circuitJson.filter(\n    (elm) => elm.type === \"simulation_voltage_probe\",\n  ) as SimulationVoltageProbe[]\n  const simulationSwitches = circuitJson\n    .filter(\n      (element) => (element as { type?: string }).type === \"simulation_switch\",\n    )\n    .map((element) => element as unknown as SimulationSwitch)\n  const simulationSwitchMap = new Map<string, SimulationSwitch>()\n\n  for (const simSwitch of simulationSwitches) {\n    if (simSwitch.source_component_id) {\n      simulationSwitchMap.set(simSwitch.source_component_id, simSwitch)\n    }\n  }\n\n  const connMap = getSourcePortConnectivityMapFromCircuitJson(circuitJson)\n\n  // Create node mapping from port connections\n  const nodeMap = new Map<string, string>()\n  const netToNodeName = new Map<string, string>()\n  let nodeCounter = 1\n\n  const probeNames = new Set<string>()\n  if (simulationProbes.length > 0) {\n    for (const probe of simulationProbes) {\n      if (probe.name) {\n        probeNames.add(probe.name)\n      }\n    }\n  }\n\n  // If there are probe names like N1, N2, make sure we don't have conflicts\n  const numericProbeNames = [...probeNames]\n    .map((name) => /^N(\\d+)$/i.exec(name))\n    .filter((m): m is RegExpExecArray => m !== null)\n    .map((m) => parseInt(m[1], 10))\n\n  if (numericProbeNames.length > 0) {\n    nodeCounter = Math.max(...numericProbeNames) + 1\n  }\n\n  const groundNets = new Set<string>()\n\n  // Find ground from source nets that include \"gnd\" in the name\n  const gndSourceNetIds = new Set(\n    su(circuitJson)\n      .source_net.list()\n      .filter((sn) => sn.name?.toLowerCase().includes(\"gnd\"))\n      .map((sn) => sn.source_net_id),\n  )\n\n  if (gndSourceNetIds.size > 0) {\n    for (const trace of su(circuitJson).source_trace.list()) {\n      if (trace.connected_source_port_ids.length > 0) {\n        const isOnGndNet = trace.connected_source_net_ids.some((netId) =>\n          gndSourceNetIds.has(netId),\n        )\n        if (isOnGndNet) {\n          const aPortOnGnd = trace.connected_source_port_ids[0]\n          const gndNet = connMap.getNetConnectedToId(aPortOnGnd)\n          if (gndNet) {\n            groundNets.add(gndNet)\n          }\n        }\n      }\n    }\n  }\n\n  // Find ground node from ports named \"GND\"\n  const groundPorts = sourcePorts.filter((p) => p.name?.toLowerCase() === \"gnd\")\n  for (const groundPort of groundPorts) {\n    const groundNet = connMap.getNetConnectedToId(groundPort.source_port_id)\n    if (groundNet) {\n      groundNets.add(groundNet)\n    }\n  }\n\n  for (const simSource of su(circuitJson).simulation_voltage_source.list()) {\n    const neg_port_id =\n      (simSource as any).negative_source_port_id ??\n      (simSource as any).terminal2_source_port_id\n    if (neg_port_id) {\n      const gnd_net = connMap.getNetConnectedToId(neg_port_id)\n      if (gnd_net) {\n        groundNets.add(gnd_net)\n      }\n    }\n  }\n\n  for (const groundNet of groundNets) {\n    netToNodeName.set(groundNet, \"0\")\n  }\n\n  // Pre-assign node names from voltage probes\n  if (simulationProbes.length > 0) {\n    for (const probe of simulationProbes) {\n      if (!probe.name) continue\n      let net: string | undefined | null\n      if (probe.source_port_id) {\n        net = connMap.getNetConnectedToId(probe.source_port_id)\n      } else if (probe.source_net_id) {\n        const trace = sourceTraces.find((t) =>\n          t.connected_source_net_ids.includes(probe.source_net_id!),\n        )\n        if (trace && trace.connected_source_port_ids.length > 0) {\n          const portId = trace.connected_source_port_ids[0]\n          net = connMap.getNetConnectedToId(portId)\n        }\n      }\n\n      if (net) {\n        if (!netToNodeName.has(net)) {\n          netToNodeName.set(net, probe.name)\n        }\n      } else if (probe.source_port_id && probe.name) {\n        // It's a floating port with a probe, so we map it directly. This port\n        // will now be skipped in the second-pass for unconnected ports.\n        nodeMap.set(probe.source_port_id, probe.name)\n      }\n    }\n  }\n\n  // First pass: assign node numbers to all connected nets\n  for (const port of sourcePorts) {\n    const portId = port.source_port_id\n    const net = connMap.getNetConnectedToId(portId)\n    if (net) {\n      if (!netToNodeName.has(net)) {\n        netToNodeName.set(net, `N${nodeCounter++}`)\n      }\n      nodeMap.set(portId, netToNodeName.get(net)!)\n    }\n  }\n\n  // Second pass: assign node numbers to unconnected ports\n  for (const port of sourcePorts) {\n    const portId = port.source_port_id\n    // If a port wasn't in a net, it won't be in the nodeMap yet\n    if (!nodeMap.has(portId)) {\n      // Unconnected port, create a new floating node for it\n      nodeMap.set(portId, `N${nodeCounter++}`)\n    }\n  }\n\n  // Process each component\n  for (const component of sourceComponents) {\n    if (component.type !== \"source_component\") continue\n\n    const componentPorts = su(circuitJson)\n      .source_port.list({\n        source_component_id: component.source_component_id,\n      })\n      .sort((a, b) => (a.pin_number ?? 0) - (b.pin_number ?? 0))\n\n    // Get node names for component ports\n    const nodes = componentPorts.map((port) => {\n      return nodeMap.get(port.source_port_id) || \"0\"\n    })\n\n    // Create SPICE component based on type\n    if (\"ftype\" in component) {\n      let spiceComponent: SpiceComponent | null = null\n\n      switch (component.ftype) {\n        case \"simple_resistor\": {\n          if (\"resistance\" in component && \"name\" in component) {\n            const resistorCmd = new ResistorCommand({\n              name: component.name,\n              positiveNode: nodes[0] || \"0\",\n              negativeNode: nodes[1] || \"0\",\n              value: formatResistance(component.resistance),\n            })\n            spiceComponent = new SpiceComponent(\n              component.name,\n              resistorCmd,\n              nodes,\n            )\n          }\n          break\n        }\n        case \"simple_switch\": {\n          const sanitizedBase = sanitizeIdentifier(\n            component.name ?? component.source_component_id,\n            \"SW\",\n          )\n          const positiveNode = nodes[0] || \"0\"\n          const negativeNode = nodes[1] || \"0\"\n          const controlNode = `NCTRL_${sanitizedBase}`\n          const modelName = `SW_${sanitizedBase}`\n\n          const associatedSimulationSwitch = simulationSwitchMap.get(\n            component.source_component_id,\n          )\n\n          const controlValue = buildSimulationSwitchControlValue(\n            associatedSimulationSwitch,\n          )\n\n          const switchCmd = new VoltageControlledSwitchCommand({\n            name: sanitizedBase,\n            positiveNode,\n            negativeNode,\n            positiveControl: controlNode,\n            negativeControl: \"0\",\n            model: modelName,\n          })\n\n          spiceComponent = new SpiceComponent(sanitizedBase, switchCmd, [\n            positiveNode,\n            negativeNode,\n            controlNode,\n            \"0\",\n          ])\n\n          if (!netlist.models.has(modelName)) {\n            netlist.models.set(\n              modelName,\n              `.MODEL ${modelName} SW(Ron=0.1 Roff=1e9 Vt=2.5 Vh=0.1)`,\n            )\n          }\n\n          const controlSourceName = `CTRL_${sanitizedBase}`\n          const controlSourceCmd = new VoltageSourceCommand({\n            name: controlSourceName,\n            positiveNode: controlNode,\n            negativeNode: \"0\",\n            value: controlValue,\n          })\n\n          const controlComponent = new SpiceComponent(\n            controlSourceName,\n            controlSourceCmd,\n            [controlNode, \"0\"],\n          )\n\n          netlist.addComponent(controlComponent)\n          break\n        }\n\n        case \"simple_capacitor\": {\n          if (\"capacitance\" in component && \"name\" in component) {\n            const capacitorCmd = new CapacitorCommand({\n              name: component.name,\n              positiveNode: nodes[0] || \"0\",\n              negativeNode: nodes[1] || \"0\",\n              value: formatCapacitance(component.capacitance),\n            })\n            spiceComponent = new SpiceComponent(\n              component.name,\n              capacitorCmd,\n              nodes,\n            )\n          }\n          break\n        }\n        case \"simple_diode\": {\n          if (\"name\" in component) {\n            const anodePort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"anode\" ||\n                p.port_hints?.includes(\"anode\"),\n            )\n            const cathodePort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"cathode\" ||\n                p.port_hints?.includes(\"cathode\"),\n            )\n            const positiveNode =\n              nodeMap.get(anodePort?.source_port_id ?? \"\") || \"0\"\n            const negativeNode =\n              nodeMap.get(cathodePort?.source_port_id ?? \"\") || \"0\"\n\n            const modelName = \"D\"\n            const diodeCmd = new DiodeCommand({\n              name: component.name,\n              positiveNode,\n              negativeNode,\n              model: modelName, // generic model\n            })\n            netlist.models.set(modelName, `.MODEL ${modelName} D`)\n            spiceComponent = new SpiceComponent(component.name, diodeCmd, [\n              positiveNode,\n              negativeNode,\n            ])\n          }\n          break\n        }\n        case \"simple_inductor\": {\n          if (\"inductance\" in component && \"name\" in component) {\n            const inductorCmd = new InductorCommand({\n              name: component.name,\n              positiveNode: nodes[0] || \"0\",\n              negativeNode: nodes[1] || \"0\",\n              value: formatInductance(component.inductance),\n            })\n            spiceComponent = new SpiceComponent(\n              component.name,\n              inductorCmd,\n              nodes,\n            )\n          }\n          break\n        }\n        case \"simple_mosfet\": {\n          if (\"name\" in component) {\n            const drainPort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"drain\" ||\n                p.port_hints?.includes(\"drain\"),\n            )\n            const gatePort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"gate\" ||\n                p.port_hints?.includes(\"gate\"),\n            )\n            const sourcePort = componentPorts.find(\n              (p) =>\n                p.name?.toLowerCase() === \"source\" ||\n                p.port_hints?.includes(\"source\"),\n            )\n\n            const drainNode =\n              nodeMap.get(drainPort?.source_port_id ?? \"\") || \"0\"\n            const gateNode = nodeMap.get(gatePort?.source_port_id ?? \"\") || \"0\"\n            const sourceNode =\n              nodeMap.get(sourcePort?.source_port_id ?? \"\") || \"0\"\n\n            const modelName = \"SWMOD\"\n            const switchCmd = new VoltageControlledSwitchCommand({\n              name: component.name,\n              positiveNode: drainNode,\n              negativeNode: sourceNode,\n              positiveControl: gateNode,\n              negativeControl: sourceNode,\n              model: modelName,\n            })\n            netlist.models.set(modelName, `.MODEL ${modelName} SW`)\n\n            spiceComponent = new SpiceComponent(component.name, switchCmd, [\n              drainNode,\n              gateNode,\n              sourceNode,\n            ])\n          }\n          break\n        }\n      }\n\n      if (spiceComponent) {\n        netlist.addComponent(spiceComponent)\n      }\n    }\n  }\n\n  // Process simulation voltage sources\n  const simulationVoltageSources =\n    su(circuitJson).simulation_voltage_source.list()\n\n  for (const simSource of simulationVoltageSources) {\n    if (simSource.type !== \"simulation_voltage_source\") continue\n\n    if ((simSource as any).is_dc_source === false) {\n      // AC Source\n      if (\n        \"terminal1_source_port_id\" in simSource &&\n        \"terminal2_source_port_id\" in simSource &&\n        (simSource as any).terminal1_source_port_id &&\n        (simSource as any).terminal2_source_port_id\n      ) {\n        const positiveNode =\n          nodeMap.get((simSource as any).terminal1_source_port_id) || \"0\"\n        const negativeNode =\n          nodeMap.get((simSource as any).terminal2_source_port_id) || \"0\"\n\n        let value = \"\"\n        const wave_shape = (simSource as any).wave_shape\n        if (wave_shape === \"sinewave\") {\n          const v_offset = 0 // not provided in circuitJson\n          const v_peak = (simSource as any).voltage ?? 0\n          const freq = (simSource as any).frequency ?? 0\n          const delay = 0 // not provided in circuitJson\n          const damping_factor = 0 // not provided in circuitJson\n          const phase = (simSource as any).phase ?? 0\n          if (freq > 0) {\n            value = `SIN(${v_offset} ${v_peak} ${freq} ${delay} ${damping_factor} ${phase})`\n          } else {\n            value = `DC ${(simSource as any).voltage ?? 0}`\n          }\n        } else if (wave_shape === \"square\") {\n          const v_initial = 0\n          const v_pulsed = (simSource as any).voltage ?? 0\n          const freq = (simSource as any).frequency ?? 0\n          const period_from_freq = freq === 0 ? Infinity : 1 / freq\n          const period = (simSource as any).period ?? period_from_freq\n          const duty_cycle = (simSource as any).duty_cycle ?? 0.5\n          const pulse_width = period * duty_cycle\n          const delay = 0\n          const rise_time = \"1n\"\n          const fall_time = \"1n\"\n          value = `PULSE(${v_initial} ${v_pulsed} ${delay} ${rise_time} ${fall_time} ${pulse_width} ${period})`\n        } else if ((simSource as any).voltage !== undefined) {\n          value = `DC ${(simSource as any).voltage}`\n        }\n\n        if (value) {\n          const voltageSourceCmd = new VoltageSourceCommand({\n            name: simSource.simulation_voltage_source_id,\n            positiveNode,\n            negativeNode,\n            value,\n          })\n\n          const spiceComponent = new SpiceComponent(\n            simSource.simulation_voltage_source_id,\n            voltageSourceCmd,\n            [positiveNode, negativeNode],\n          )\n          netlist.addComponent(spiceComponent)\n        }\n      }\n    } else {\n      // DC Source (is_dc_source is true or undefined)\n      const positivePortId =\n        (simSource as any).positive_source_port_id ??\n        (simSource as any).terminal1_source_port_id\n      const negativePortId =\n        (simSource as any).negative_source_port_id ??\n        (simSource as any).terminal2_source_port_id\n\n      if (\n        positivePortId &&\n        negativePortId &&\n        \"voltage\" in simSource &&\n        (simSource as any).voltage !== undefined\n      ) {\n        const positiveNode = nodeMap.get(positivePortId) || \"0\"\n        const negativeNode = nodeMap.get(negativePortId) || \"0\"\n\n        const voltageSourceCmd = new VoltageSourceCommand({\n          name: simSource.simulation_voltage_source_id,\n          positiveNode,\n          negativeNode,\n          value: `DC ${(simSource as any).voltage}`,\n        })\n\n        const spiceComponent = new SpiceComponent(\n          simSource.simulation_voltage_source_id,\n          voltageSourceCmd,\n          [positiveNode, negativeNode],\n        )\n        netlist.addComponent(spiceComponent)\n      }\n    }\n  }\n\n  const simExperiment = circuitJson.find(\n    (elm) => elm.type === \"simulation_experiment\",\n  )\n\n  if (simExperiment) {\n    // Process simulation voltage probes\n    if (simulationProbes.length > 0) {\n      const nodesToProbe = new Set<string>()\n\n      for (const probe of simulationProbes) {\n        let nodeName: string | undefined\n        if (probe.source_port_id) {\n          nodeName = nodeMap.get(probe.source_port_id)\n        } else if (probe.source_net_id) {\n          const trace = sourceTraces.find((t) =>\n            t.connected_source_net_ids.includes(probe.source_net_id!),\n          )\n          if (trace && trace.connected_source_port_ids.length > 0) {\n            const portId = trace.connected_source_port_ids[0]\n            nodeName = nodeMap.get(portId)\n          }\n        }\n\n        if (nodeName && nodeName !== \"0\") {\n          nodesToProbe.add(`V(${nodeName})`)\n        }\n      }\n\n      if (\n        nodesToProbe.size > 0 &&\n        (simExperiment as any).experiment_type?.includes(\"transient\")\n      ) {\n        netlist.printStatements.push(\n          `.PRINT TRAN ${[...nodesToProbe].join(\" \")}`,\n        )\n      }\n    }\n\n    const timePerStep = (simExperiment as any).time_per_step\n    const endTime = (simExperiment as any).end_time_ms\n    const startTimeMs = (simExperiment as any).start_time_ms\n\n    if (timePerStep && endTime) {\n      // circuit-json values are in ms, SPICE requires seconds\n      const startTime = (startTimeMs ?? 0) / 1000\n\n      let tranCmd = `.tran ${formatNumberForSpice(\n        timePerStep / 1000,\n      )} ${formatNumberForSpice(endTime / 1000)}`\n      if (startTime > 0) {\n        tranCmd += ` ${formatNumberForSpice(startTime)}`\n      }\n      tranCmd += \" UIC\"\n      netlist.tranCommand = tranCmd\n    }\n  }\n\n  return netlist\n}\n\nfunction formatResistance(resistance: number): string {\n  if (resistance >= 1e6) return `${resistance / 1e6}MEG`\n  if (resistance >= 1e3) return `${resistance / 1e3}K`\n  return resistance.toString()\n}\n\nfunction formatCapacitance(capacitance: number): string {\n  if (capacitance >= 1e-3) return `${capacitance * 1e3}M`\n  if (capacitance >= 1e-6) return `${capacitance * 1e6}U`\n  if (capacitance >= 1e-9) return `${capacitance * 1e9}N`\n  if (capacitance >= 1e-12) return `${capacitance * 1e12}P`\n  return capacitance.toString()\n}\n\nfunction formatInductance(inductance: number): string {\n  if (inductance >= 1) return inductance.toString()\n  if (inductance >= 1e-3) return `${inductance * 1e3}m`\n  if (inductance >= 1e-6) return `${inductance * 1e6}u`\n  if (inductance >= 1e-9) return `${inductance * 1e9}n`\n  if (inductance >= 1e-12) return `${inductance * 1e12}p`\n  return inductance.toString()\n}\n\nfunction sanitizeIdentifier(value: string | undefined, prefix: string) {\n  if (!value) return prefix\n  const sanitized = value.replace(/[^A-Za-z0-9_]/g, \"_\")\n  if (!sanitized) return prefix\n  if (/^[0-9]/.test(sanitized)) {\n    return `${prefix}_${sanitized}`\n  }\n  return sanitized\n}\n\nfunction buildSimulationSwitchControlValue(\n  simulationSwitch: SimulationSwitch | undefined,\n) {\n  const highVoltage = 5\n  const lowVoltage = 0\n  const riseTime = \"1n\"\n  const fallTime = \"1n\"\n\n  if (!simulationSwitch) {\n    return `DC ${lowVoltage}`\n  }\n\n  const startsClosed = simulationSwitch.starts_closed ?? false\n  const closesAt = simulationSwitch.closes_at ?? 0\n  const opensAt = simulationSwitch.opens_at\n  const switchingFrequency = simulationSwitch.switching_frequency\n\n  const [initialVoltage, pulsedVoltage] = startsClosed\n    ? [highVoltage, lowVoltage]\n    : [lowVoltage, highVoltage]\n\n  if (switchingFrequency && switchingFrequency > 0) {\n    const period = 1 / switchingFrequency\n    const widthFromOpenClose =\n      opensAt && opensAt > closesAt ? Math.min(opensAt - closesAt, period) : 0\n    const pulseWidth =\n      widthFromOpenClose > 0 ? widthFromOpenClose : Math.max(period / 2, 1e-9)\n\n    return `PULSE(${formatNumberForSpice(initialVoltage)} ${formatNumberForSpice(pulsedVoltage)} ${formatNumberForSpice(closesAt)} ${riseTime} ${fallTime} ${formatNumberForSpice(pulseWidth)} ${formatNumberForSpice(period)})`\n  }\n\n  if (opensAt !== undefined && opensAt > closesAt) {\n    const pulseWidth = Math.max(opensAt - closesAt, 1e-9)\n    const period = closesAt + pulseWidth * 2\n\n    return `PULSE(${formatNumberForSpice(initialVoltage)} ${formatNumberForSpice(pulsedVoltage)} ${formatNumberForSpice(closesAt)} ${riseTime} ${fallTime} ${formatNumberForSpice(pulseWidth)} ${formatNumberForSpice(period)})`\n  }\n\n  if (closesAt > 0) {\n    const period = closesAt * 2\n    const pulseWidth = Math.max(period / 2, 1e-9)\n    return `PULSE(${formatNumberForSpice(initialVoltage)} ${formatNumberForSpice(pulsedVoltage)} ${formatNumberForSpice(closesAt)} ${riseTime} ${fallTime} ${formatNumberForSpice(pulseWidth)} ${formatNumberForSpice(period)})`\n  }\n\n  return `DC ${startsClosed ? highVoltage : lowVoltage}`\n}\n\nfunction formatNumberForSpice(value: number) {\n  if (!Number.isFinite(value)) return `${value}`\n  if (value === 0) return \"0\"\n\n  const absValue = Math.abs(value)\n\n  if (absValue >= 1e3 || absValue <= 1e-3) {\n    return Number(value.toExponential(6)).toString()\n  }\n\n  return Number(value.toPrecision(6)).toString()\n}\n", "export class SpiceSubcircuit {\n  name: string\n  pins: string[]\n\n  constructor(name: string, pins: string[]) {\n    this.name = name\n    this.pins = pins\n  }\n\n  toSpiceString(): string {\n    const pinString = this.pins.join(\" \")\n    const header = `.SUBCKT ${this.name} ${pinString}`\n    const footer = `.ENDS ${this.name}`\n\n    const body = `* Placeholder for ${this.name}. No definition found in circuit JSON.`\n\n    return [\"\", header, body, footer].join(\"\\n\")\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface BJTCommandProps {\n  name: string\n  collector: string\n  base: string\n  emitter: string\n  substrate?: string\n  model: string\n  area?: string\n}\n\nexport class BJTCommand implements BaseSpiceCommand {\n  commandName = \"bjt\" as const\n  props: BJTCommandProps\n\n  constructor(props: BJTCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, collector, base, emitter, substrate, model, area } =\n      this.props\n    let spiceString = `Q${name} ${collector} ${base} ${emitter}`\n    if (substrate) {\n      spiceString += ` ${substrate}`\n    }\n    spiceString += ` ${model}`\n    if (area) {\n      spiceString += ` ${area}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface CurrentSourceCommandProps {\n  name: string\n  positiveNode: string\n  negativeNode: string\n  value?: string\n  acMagnitude?: string\n  acPhase?: string\n}\n\nexport class CurrentSourceCommand implements BaseSpiceCommand {\n  commandName = \"current_source\" as const\n  props: CurrentSourceCommandProps\n\n  constructor(props: CurrentSourceCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, positiveNode, negativeNode, value, acMagnitude, acPhase } =\n      this.props\n    let spiceString = `I${name} ${positiveNode} ${negativeNode}`\n    if (value) {\n      spiceString += ` ${value}`\n    }\n    if (acMagnitude) {\n      spiceString += ` AC ${acMagnitude}`\n      if (acPhase) {\n        spiceString += ` ${acPhase}`\n      }\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface InductorCouplingCommandProps {\n  name: string\n  inductors: string[]\n  coupling: string\n}\n\nexport class InductorCouplingCommand implements BaseSpiceCommand {\n  commandName = \"inductor_coupling\" as const\n  props: InductorCouplingCommandProps\n\n  constructor(props: InductorCouplingCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, inductors, coupling } = this.props\n    return `K${name} ${inductors.join(\" \")} ${coupling}`\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface JFETCommandProps {\n  name: string\n  drain: string\n  gate: string\n  source: string\n  model: string\n  area?: string\n}\n\nexport class JFETCommand implements BaseSpiceCommand {\n  commandName = \"jfet\" as const\n  props: JFETCommandProps\n\n  constructor(props: JFETCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, drain, gate, source, model, area } = this.props\n    let spiceString = `J${name} ${drain} ${gate} ${source} ${model}`\n    if (area) {\n      spiceString += ` ${area}`\n    }\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface MOSFETCommandProps {\n  name: string\n  drain: string\n  gate: string\n  source: string\n  substrate: string\n  model: string\n  length?: string\n  width?: string\n  drainArea?: string\n  sourceArea?: string\n  drainPerimeter?: string\n  sourcePerimeter?: string\n  drainResistance?: string\n  sourceResistance?: string\n}\n\nexport class MOSFETCommand implements BaseSpiceCommand {\n  commandName = \"mosfet\" as const\n  props: MOSFETCommandProps\n\n  constructor(props: MOSFETCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const {\n      name,\n      drain,\n      gate,\n      source,\n      substrate,\n      model,\n      length,\n      width,\n      drainArea,\n      sourceArea,\n      drainPerimeter,\n      sourcePerimeter,\n      drainResistance,\n      sourceResistance,\n    } = this.props\n\n    let spiceString = `M${name} ${drain} ${gate} ${source} ${substrate} ${model}`\n\n    const params: Record<string, string | undefined> = {\n      L: length,\n      W: width,\n      AD: drainArea,\n      AS: sourceArea,\n      PD: drainPerimeter,\n      PS: sourcePerimeter,\n      NRD: drainResistance,\n      NRS: sourceResistance,\n    }\n\n    Object.entries(params).forEach(([key, value]) => {\n      if (value) {\n        spiceString += ` ${key}=${value}`\n      }\n    })\n\n    return spiceString\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface SubcircuitCallCommandProps {\n  name: string\n  nodes: string[]\n  subcircuitName: string\n}\n\nexport class SubcircuitCallCommand implements BaseSpiceCommand {\n  commandName = \"subcircuit_call\" as const\n  props: SubcircuitCallCommandProps\n\n  constructor(props: SubcircuitCallCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const { name, nodes, subcircuitName } = this.props\n    return `X${name} ${nodes.join(\" \")} ${subcircuitName}`\n  }\n}\n", "import type { BaseSpiceCommand } from \"./BaseSpiceCommand\"\n\nexport interface TransmissionLineCommandProps {\n  name: string\n  aPositive: string\n  aNegative: string\n  bPositive: string\n  bNegative: string\n  impedance: string\n  delay?: string\n  frequency?: string\n  normalizedLength?: string\n}\n\nexport class TransmissionLineCommand implements BaseSpiceCommand {\n  commandName = \"transmission_line\" as const\n  props: TransmissionLineCommandProps\n\n  constructor(props: TransmissionLineCommandProps) {\n    this.props = props\n  }\n\n  toSpiceString(): string {\n    const {\n      name,\n      aPositive,\n      aNegative,\n      bPositive,\n      bNegative,\n      impedance,\n      delay,\n      frequency,\n      normalizedLength,\n    } = this.props\n    let spiceString = `T${name} ${aPositive} ${aNegative} ${bPositive} ${bNegative} Z0=${impedance}`\n    if (delay) {\n      spiceString += ` TD=${delay}`\n    } else if (frequency) {\n      spiceString += ` F=${frequency}`\n      if (normalizedLength) {\n        spiceString += ` NL=${normalizedLength}`\n      }\n    }\n    return spiceString\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

@@ -6,7 +6,11 @@ import { VoltageSourceCommand } from "./spice-commands/VoltageSourceCommand"
6
6
  import { DiodeCommand } from "./spice-commands/DiodeCommand"
7
7
  import { InductorCommand } from "./spice-commands/InductorCommand"
8
8
  import { VoltageControlledSwitchCommand } from "./spice-commands/VoltageControlledSwitchCommand"
9
- import type { AnyCircuitElement, SimulationSwitch } from "circuit-json"
9
+ import type {
10
+ AnyCircuitElement,
11
+ SimulationSwitch,
12
+ SimulationVoltageProbe,
13
+ } from "circuit-json"
10
14
  import { getSourcePortConnectivityMapFromCircuitJson } from "circuit-json-to-connectivity-map"
11
15
  import { su } from "@tscircuit/circuit-json-util"
12
16
 
@@ -16,6 +20,10 @@ export function circuitJsonToSpice(
16
20
  const netlist = new SpiceNetlist("* Circuit JSON to SPICE Netlist")
17
21
  const sourceComponents = su(circuitJson).source_component.list()
18
22
  const sourcePorts = su(circuitJson).source_port.list()
23
+ const sourceTraces = su(circuitJson).source_trace.list()
24
+ const simulationProbes = circuitJson.filter(
25
+ (elm) => elm.type === "simulation_voltage_probe",
26
+ ) as SimulationVoltageProbe[]
19
27
  const simulationSwitches = circuitJson
20
28
  .filter(
21
29
  (element) => (element as { type?: string }).type === "simulation_switch",
@@ -36,6 +44,25 @@ export function circuitJsonToSpice(
36
44
  const netToNodeName = new Map<string, string>()
37
45
  let nodeCounter = 1
38
46
 
47
+ const probeNames = new Set<string>()
48
+ if (simulationProbes.length > 0) {
49
+ for (const probe of simulationProbes) {
50
+ if (probe.name) {
51
+ probeNames.add(probe.name)
52
+ }
53
+ }
54
+ }
55
+
56
+ // If there are probe names like N1, N2, make sure we don't have conflicts
57
+ const numericProbeNames = [...probeNames]
58
+ .map((name) => /^N(\d+)$/i.exec(name))
59
+ .filter((m): m is RegExpExecArray => m !== null)
60
+ .map((m) => parseInt(m[1], 10))
61
+
62
+ if (numericProbeNames.length > 0) {
63
+ nodeCounter = Math.max(...numericProbeNames) + 1
64
+ }
65
+
39
66
  const groundNets = new Set<string>()
40
67
 
41
68
  // Find ground from source nets that include "gnd" in the name
@@ -88,6 +115,35 @@ export function circuitJsonToSpice(
88
115
  netToNodeName.set(groundNet, "0")
89
116
  }
90
117
 
118
+ // Pre-assign node names from voltage probes
119
+ if (simulationProbes.length > 0) {
120
+ for (const probe of simulationProbes) {
121
+ if (!probe.name) continue
122
+ let net: string | undefined | null
123
+ if (probe.source_port_id) {
124
+ net = connMap.getNetConnectedToId(probe.source_port_id)
125
+ } else if (probe.source_net_id) {
126
+ const trace = sourceTraces.find((t) =>
127
+ t.connected_source_net_ids.includes(probe.source_net_id!),
128
+ )
129
+ if (trace && trace.connected_source_port_ids.length > 0) {
130
+ const portId = trace.connected_source_port_ids[0]
131
+ net = connMap.getNetConnectedToId(portId)
132
+ }
133
+ }
134
+
135
+ if (net) {
136
+ if (!netToNodeName.has(net)) {
137
+ netToNodeName.set(net, probe.name)
138
+ }
139
+ } else if (probe.source_port_id && probe.name) {
140
+ // It's a floating port with a probe, so we map it directly. This port
141
+ // will now be skipped in the second-pass for unconnected ports.
142
+ nodeMap.set(probe.source_port_id, probe.name)
143
+ }
144
+ }
145
+ }
146
+
91
147
  // First pass: assign node numbers to all connected nets
92
148
  for (const port of sourcePorts) {
93
149
  const portId = port.source_port_id
@@ -427,16 +483,7 @@ export function circuitJsonToSpice(
427
483
 
428
484
  if (simExperiment) {
429
485
  // Process simulation voltage probes
430
- const simulationProbes = circuitJson.filter(
431
- (elm) => elm.type === "simulation_voltage_probe",
432
- ) as unknown as Array<{
433
- type: "simulation_voltage_probe"
434
- source_port_id?: string
435
- source_net_id?: string
436
- }>
437
-
438
486
  if (simulationProbes.length > 0) {
439
- const sourceTraces = su(circuitJson).source_trace.list()
440
487
  const nodesToProbe = new Set<string>()
441
488
 
442
489
  for (const probe of simulationProbes) {
@@ -454,7 +501,7 @@ export function circuitJsonToSpice(
454
501
  }
455
502
 
456
503
  if (nodeName && nodeName !== "0") {
457
- nodesToProbe.add(`V(${nodeName.toLowerCase()})`)
504
+ nodesToProbe.add(`V(${nodeName})`)
458
505
  }
459
506
  }
460
507
 
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "circuit-json-to-spice",
3
3
  "main": "dist/index.js",
4
- "version": "0.0.22",
4
+ "version": "0.0.24",
5
5
  "type": "module",
6
6
  "scripts": {
7
7
  "build": "tsup-node ./lib/index.ts --dts --format esm --sourcemap inline -d dist",
@@ -2,36 +2,40 @@ import { test, expect } from "bun:test"
2
2
  import { getTestFixture } from "tests/fixtures/getTestFixture"
3
3
  import { circuitJsonToSpice } from "lib/circuitJsonToSpice"
4
4
 
5
- test("AC voltage source", async () => {
6
- const { circuit } = await getTestFixture()
5
+ test(
6
+ "AC voltage source",
7
+ async () => {
8
+ const { circuit } = await getTestFixture()
7
9
 
8
- circuit.add(
9
- <board width="10mm" height="10mm">
10
- <voltagesource
11
- name="VS1"
12
- voltage="5V"
13
- frequency="60Hz"
14
- waveShape="sinewave"
15
- />
16
- <resistor name="R1" resistance="10k" pcbY={-2} schY={-2} />
17
- <resistor name="R2" resistance="10k" pcbY={-2} schY={-2} />
18
- <trace from={".VS1 > .pin1"} to={".R1 > .pin1"} />
19
- <trace from={".R1 > .pin2"} to={".R2 > .pin1"} />
20
- <trace from={".VS1 > .pin2"} to={".R2 > .pin2"} />
21
- </board>,
22
- )
10
+ circuit.add(
11
+ <board width="10mm" height="10mm">
12
+ <voltagesource
13
+ name="VS1"
14
+ voltage="5V"
15
+ frequency="60Hz"
16
+ waveShape="sinewave"
17
+ />
18
+ <resistor name="R1" resistance="10k" pcbY={-2} schY={-2} />
19
+ <resistor name="R2" resistance="10k" pcbY={2} schY={2} />
20
+ <trace from={".VS1 > .pin1"} to={".R1 > .pin1"} />
21
+ <trace from={".R1 > .pin2"} to={".R2 > .pin1"} />
22
+ <trace from={".VS1 > .pin2"} to={".R2 > .pin2"} />
23
+ </board>,
24
+ )
23
25
 
24
- await circuit.renderUntilSettled()
25
- const circuitJson = circuit.getCircuitJson()
26
+ await circuit.renderUntilSettled()
27
+ const circuitJson = circuit.getCircuitJson()
26
28
 
27
- const spiceNetlist = circuitJsonToSpice(circuitJson)
28
- const spiceString = spiceNetlist.toSpiceString()
29
+ const spiceNetlist = circuitJsonToSpice(circuitJson)
30
+ const spiceString = spiceNetlist.toSpiceString()
29
31
 
30
- expect(spiceString).toMatchInlineSnapshot(`
32
+ expect(spiceString).toMatchInlineSnapshot(`
31
33
  "* Circuit JSON to SPICE Netlist
32
34
  RR1 N1 N2 10K
33
35
  RR2 N2 0 10K
34
36
  Vsimulation_voltage_source_0 N1 0 SIN(0 5 60 0 0 0)
35
37
  .END"
36
38
  `)
37
- })
39
+ },
40
+ { timeout: 10000 },
41
+ )
@@ -1,5 +1,5 @@
1
1
  import { expect, test } from "bun:test"
2
- import type { AnyCircuitElement } from "circuit-json"
2
+ import type { AnyCircuitElement, SimulationVoltageProbe } from "circuit-json"
3
3
  import { circuitJsonToSpice } from "lib/circuitJsonToSpice"
4
4
 
5
5
  // A simple resistor divider circuit
@@ -16,12 +16,14 @@ const baseCircuit: AnyCircuitElement[] = [
16
16
  source_port_id: "R1_p1",
17
17
  source_component_id: "R1",
18
18
  name: "p1",
19
+ pin_number: 1,
19
20
  },
20
21
  {
21
22
  type: "source_port",
22
23
  source_port_id: "R1_p2",
23
24
  source_component_id: "R1",
24
25
  name: "p2",
26
+ pin_number: 2,
25
27
  },
26
28
  {
27
29
  type: "source_component",
@@ -35,12 +37,14 @@ const baseCircuit: AnyCircuitElement[] = [
35
37
  source_port_id: "R2_p1",
36
38
  source_component_id: "R2",
37
39
  name: "p1",
40
+ pin_number: 1,
38
41
  },
39
42
  {
40
43
  type: "source_port",
41
44
  source_port_id: "R2_p2",
42
45
  source_component_id: "R2",
43
46
  name: "p2",
47
+ pin_number: 2,
44
48
  },
45
49
  {
46
50
  type: "source_trace",
@@ -70,18 +74,16 @@ test("voltage probe with source_port_id creates .PRINT statement", () => {
70
74
  {
71
75
  type: "simulation_voltage_probe",
72
76
  source_port_id: "R1_p2",
73
- name: "probe1",
74
- } as any,
77
+ name: "VOUT",
78
+ } as SimulationVoltageProbe,
75
79
  ]
76
80
 
77
81
  const netlist = circuitJsonToSpice(circuitJson)
78
82
  const spiceString = netlist.toSpiceString()
79
83
 
80
- const r1Match = /RR1 (N\d+) (N\d+)/.exec(spiceString)
81
- expect(r1Match).not.toBeNull()
82
- const r1p2Node = r1Match![2]
83
-
84
- expect(spiceString).toContain(`.PRINT TRAN V(${r1p2Node.toLowerCase()})`)
84
+ expect(spiceString).toContain(`.PRINT TRAN V(VOUT)`)
85
+ expect(spiceString).toContain(`RR1 N1 VOUT 1K`)
86
+ expect(spiceString).toContain(`RR2 VOUT N2 1K`)
85
87
  })
86
88
 
87
89
  test("voltage probe with source_net_id creates .PRINT statement", () => {
@@ -90,18 +92,16 @@ test("voltage probe with source_net_id creates .PRINT statement", () => {
90
92
  {
91
93
  type: "simulation_voltage_probe",
92
94
  source_net_id: "net1",
93
- name: "probe2",
94
- } as any,
95
+ name: "VOUT",
96
+ } as SimulationVoltageProbe,
95
97
  ]
96
98
 
97
99
  const netlist = circuitJsonToSpice(circuitJson)
98
100
  const spiceString = netlist.toSpiceString()
99
101
 
100
- const r1Match = /RR1 (N\d+) (N\d+)/.exec(spiceString)
101
- expect(r1Match).not.toBeNull()
102
- const net1Node = r1Match![2]
103
-
104
- expect(spiceString).toContain(`.PRINT TRAN V(${net1Node.toLowerCase()})`)
102
+ expect(spiceString).toContain(`.PRINT TRAN V(VOUT)`)
103
+ expect(spiceString).toContain(`RR1 N1 VOUT 1K`)
104
+ expect(spiceString).toContain(`RR2 VOUT N2 1K`)
105
105
  })
106
106
 
107
107
  test("voltage probe without transient analysis does not create .PRINT statement", () => {
@@ -110,14 +110,15 @@ test("voltage probe without transient analysis does not create .PRINT statement"
110
110
  {
111
111
  type: "simulation_voltage_probe",
112
112
  source_port_id: "R1_p2",
113
- name: "probe1",
114
- } as any,
113
+ name: "VOUT",
114
+ } as SimulationVoltageProbe,
115
115
  ]
116
116
 
117
117
  const netlist = circuitJsonToSpice(circuitJson)
118
118
  const spiceString = netlist.toSpiceString()
119
119
 
120
120
  expect(spiceString).not.toContain(".PRINT")
121
+ expect(spiceString).toContain("RR1 N1 VOUT 1K")
121
122
  })
122
123
 
123
124
  test("voltage probe on ground node is ignored, but other probes are not", () => {
@@ -139,25 +140,23 @@ test("voltage probe on ground node is ignored, but other probes are not", () =>
139
140
  type: "simulation_voltage_probe",
140
141
  source_port_id: "R2_p2",
141
142
  name: "probe_gnd",
142
- } as any,
143
+ } as SimulationVoltageProbe,
143
144
  {
144
145
  type: "simulation_voltage_probe",
145
146
  source_port_id: "R1_p2",
146
- name: "probe_non_gnd",
147
- } as any,
147
+ name: "VOUT",
148
+ } as SimulationVoltageProbe,
148
149
  ]
149
150
 
150
151
  const netlist = circuitJsonToSpice(circuitJson)
151
152
  const spiceString = netlist.toSpiceString()
152
153
 
153
- const r1Match = /RR1 (N\d+) (N\d+)/.exec(spiceString)
154
- expect(r1Match).not.toBeNull()
155
- const r1p2Node = r1Match![2]
156
-
157
154
  // `R2_p2` is now connected to GND, so it's node 0 and should be ignored
158
155
  // `R1_p2` is a non-gnd node and should be printed.
159
- expect(spiceString).toContain(`.PRINT TRAN V(${r1p2Node.toLowerCase()})`)
156
+ expect(spiceString).toContain(`.PRINT TRAN V(VOUT)`)
160
157
  expect(spiceString).not.toContain("V(0)")
158
+ expect(spiceString).toContain("RR1 N1 VOUT 1K")
159
+ expect(spiceString).toContain("RR2 VOUT 0 1K")
161
160
  })
162
161
 
163
162
  test("multiple voltage probes create single .PRINT statement", () => {
@@ -166,25 +165,211 @@ test("multiple voltage probes create single .PRINT statement", () => {
166
165
  {
167
166
  type: "simulation_voltage_probe",
168
167
  source_port_id: "R1_p1",
169
- name: "probe1",
170
- } as any,
168
+ name: "N1",
169
+ } as SimulationVoltageProbe,
171
170
  {
172
171
  type: "simulation_voltage_probe",
173
172
  source_port_id: "R1_p2",
174
- name: "probe2",
175
- } as any,
173
+ name: "VOUT",
174
+ } as SimulationVoltageProbe,
175
+ ]
176
+
177
+ const netlist = circuitJsonToSpice(circuitJson)
178
+ const spiceString = netlist.toSpiceString()
179
+
180
+ // R1_p1 is probed as "N1", R1_p2 probed as "VOUT"
181
+ expect(spiceString).toContain("RR1 N1 VOUT 1K")
182
+ expect(spiceString).toContain("RR2 VOUT N2 1K")
183
+
184
+ // The order of probes in the .PRINT statement is not guaranteed
185
+ expect(spiceString).toContain(".PRINT TRAN")
186
+ expect(spiceString).toContain("V(N1)")
187
+ expect(spiceString).toContain("V(VOUT)")
188
+ })
189
+
190
+ test("probes with N-style names don't conflict with auto-generated names", () => {
191
+ const circuitJson: AnyCircuitElement[] = [
192
+ // R1
193
+ {
194
+ type: "source_component",
195
+ source_component_id: "R1",
196
+ name: "R1",
197
+ ftype: "simple_resistor",
198
+ resistance: 1000,
199
+ },
200
+ {
201
+ type: "source_port",
202
+ source_port_id: "R1_p1",
203
+ source_component_id: "R1",
204
+ name: "p1",
205
+ pin_number: 1,
206
+ },
207
+ {
208
+ type: "source_port",
209
+ source_port_id: "R1_p2",
210
+ source_component_id: "R1",
211
+ name: "p2",
212
+ pin_number: 2,
213
+ },
214
+ // R2
215
+ {
216
+ type: "source_component",
217
+ source_component_id: "R2",
218
+ name: "R2",
219
+ ftype: "simple_resistor",
220
+ resistance: 1000,
221
+ },
222
+ {
223
+ type: "source_port",
224
+ source_port_id: "R2_p1",
225
+ source_component_id: "R2",
226
+ name: "p1",
227
+ pin_number: 1,
228
+ },
229
+ {
230
+ type: "source_port",
231
+ source_port_id: "R2_p2",
232
+ source_component_id: "R2",
233
+ name: "p2",
234
+ pin_number: 2,
235
+ },
236
+ // R3
237
+ {
238
+ type: "source_component",
239
+ source_component_id: "R3",
240
+ name: "R3",
241
+ ftype: "simple_resistor",
242
+ resistance: 1000,
243
+ },
244
+ {
245
+ type: "source_port",
246
+ source_port_id: "R3_p1",
247
+ source_component_id: "R3",
248
+ name: "p1",
249
+ pin_number: 1,
250
+ },
251
+ {
252
+ type: "source_port",
253
+ source_port_id: "R3_p2",
254
+ source_component_id: "R3",
255
+ name: "p2",
256
+ pin_number: 2,
257
+ },
258
+ // R4
259
+ {
260
+ type: "source_component",
261
+ source_component_id: "R4",
262
+ name: "R4",
263
+ ftype: "simple_resistor",
264
+ resistance: 1000,
265
+ },
266
+ {
267
+ type: "source_port",
268
+ source_port_id: "R4_p1",
269
+ source_component_id: "R4",
270
+ name: "p1",
271
+ pin_number: 1,
272
+ },
273
+ {
274
+ type: "source_port",
275
+ source_port_id: "R4_p2",
276
+ source_component_id: "R4",
277
+ name: "p2",
278
+ pin_number: 2,
279
+ },
280
+ // R5
281
+ {
282
+ type: "source_component",
283
+ source_component_id: "R5",
284
+ name: "R5",
285
+ ftype: "simple_resistor",
286
+ resistance: 1000,
287
+ },
288
+ {
289
+ type: "source_port",
290
+ source_port_id: "R5_p1",
291
+ source_component_id: "R5",
292
+ name: "p1",
293
+ pin_number: 1,
294
+ },
295
+ {
296
+ type: "source_port",
297
+ source_port_id: "R5_p2",
298
+ source_component_id: "R5",
299
+ name: "p2",
300
+ pin_number: 2,
301
+ },
302
+
303
+ // Traces
304
+ {
305
+ type: "source_trace",
306
+ source_trace_id: "t1",
307
+ connected_source_port_ids: ["R1_p2", "R2_p1"],
308
+ connected_source_net_ids: [],
309
+ },
310
+ {
311
+ type: "source_trace",
312
+ source_trace_id: "t2",
313
+ connected_source_port_ids: ["R2_p2", "R3_p1"],
314
+ connected_source_net_ids: [],
315
+ },
316
+ {
317
+ type: "source_trace",
318
+ source_trace_id: "t3",
319
+ connected_source_port_ids: ["R3_p2", "R4_p1"],
320
+ connected_source_net_ids: [],
321
+ },
322
+ {
323
+ type: "source_trace",
324
+ source_trace_id: "t4",
325
+ connected_source_port_ids: ["R4_p2", "R5_p1"],
326
+ connected_source_net_ids: [],
327
+ },
328
+
329
+ // Sim experiment
330
+ {
331
+ type: "simulation_experiment",
332
+ simulation_experiment_id: "se1",
333
+ name: "Test Transient Analysis",
334
+ experiment_type: "spice_transient_analysis" as const,
335
+ time_per_step: 1, // ms
336
+ end_time_ms: 100, // ms
337
+ },
338
+
339
+ // Probes
340
+ {
341
+ type: "simulation_voltage_probe",
342
+ source_port_id: "R1_p1",
343
+ name: "N1",
344
+ } as SimulationVoltageProbe,
345
+ {
346
+ type: "simulation_voltage_probe",
347
+ source_port_id: "R2_p1",
348
+ name: "N2",
349
+ } as SimulationVoltageProbe,
350
+ {
351
+ type: "simulation_voltage_probe",
352
+ source_port_id: "R4_p1",
353
+ name: "N4",
354
+ } as SimulationVoltageProbe,
176
355
  ]
177
356
 
178
357
  const netlist = circuitJsonToSpice(circuitJson)
179
358
  const spiceString = netlist.toSpiceString()
180
359
 
181
- const r1Match = /RR1 (N\d+) (N\d+)/.exec(spiceString)
182
- expect(r1Match).not.toBeNull()
183
- const r1p1Node = r1Match![1]
184
- const r1p2Node = r1Match![2]
360
+ // Due to probe naming, node counting for automatic names should start after
361
+ // the max N-value from probes. Probes: N1, N2, N4. Counter should start at 5.
362
+ expect(spiceString).toContain("RR1 N1 N2 1K")
363
+ expect(spiceString).toContain("RR2 N2 N5 1K")
364
+ expect(spiceString).toContain("RR3 N5 N4 1K")
365
+ expect(spiceString).toContain("RR4 N4 N6 1K")
366
+ expect(spiceString).toContain("RR5 N6 N7 1K")
185
367
 
186
- // R1_p1 and R1_p2 are probed
187
- expect(spiceString).toContain(
188
- `.PRINT TRAN V(${r1p1Node.toLowerCase()}) V(${r1p2Node.toLowerCase()})`,
189
- )
368
+ // The order of probes in the .PRINT statement is not guaranteed
369
+ expect(spiceString).toContain(".PRINT TRAN")
370
+ expect(spiceString).toContain("V(N1)")
371
+ expect(spiceString).toContain("V(N2)")
372
+ expect(spiceString).toContain("V(N4)")
373
+ expect(spiceString).not.toContain("V(n3)")
374
+ expect(spiceString).not.toContain("V(n5)")
190
375
  })