diff --git a/src/descriptions/familySummaries.js b/src/descriptions/familySummaries.js
index d59f952dc45cfa50792a1f10b748f1a1b828fe89..7aa316de819147d93b2276476471889d3846232d 100644
--- a/src/descriptions/familySummaries.js
+++ b/src/descriptions/familySummaries.js
@@ -7,15 +7,25 @@ App.Desc.family = (function() {
 		return slaveList.map(s => s.slaveName).reduce((res, ch, i, arr) => res + (i === arr.length - 1 ? ' and ' : ', ') + ch);
 	}
 
+	/** See if an ID refers to a valid, active slave.  This will return false for slaves that no longer exist or have not yet been created.
+	 * @param {number} slaveID
+	 * @returns {boolean}
+	 */
+	function validSlaveID(slaveID) {
+		return slaveID > 0 && _.isObject(getSlave(slaveID));
+	}
+
 	/** From a slave ID, return text describing that slave (even if they aren't currently your slave) to use in place of their name
 	 * @param {number} slaveID
 	 * @returns {string}
 	 */
 	function knownSlave(slaveID) {
-		if (slaveID > 0) {
+		if (validSlaveID(slaveID)) {
 			return getSlave(slaveID).slaveName;
-		} else {
+		} else if (slaveID in V.missingTable) {
 			return "your former slave " + V.missingTable[slaveID].slaveName;
+		} else {
+			return "another slave";
 		}
 	}
 
diff --git a/src/gui/multipleInspect.js b/src/gui/multipleInspect.js
new file mode 100644
index 0000000000000000000000000000000000000000..1abd7a86b0d7691d773d153d345dc44ba7226ddd
--- /dev/null
+++ b/src/gui/multipleInspect.js
@@ -0,0 +1,68 @@
+App.UI.MultipleInspect = (function() {
+	function _LSDDOM(slave, saleDescription, applyLaw) {
+		// LSD will reset these globals, so we have to set them every single time
+		V.saleDescription = (saleDescription ? 1 : 0);
+		V.applyLaw = (applyLaw ? 1 : 0);
+
+		// embedded sugarcube sucks but that's just how LSD works right now
+		const oldAS = V.activeSlave;
+		V.activeSlave = slave;
+		const frag = document.createDocumentFragment();
+		$(frag).wiki(`<<include "Long Slave Description">>`);
+		V.activeSlave = oldAS;
+		return frag;
+	}
+
+	/**
+	 * Provide a mechanism to inspect multiple slaves at once (for example, for Household Liquidators and recETS).
+	 * Intended for use from DOM passages.
+	 * @param {Array<App.Entity.SlaveState>} slaves
+	 * @param {boolean} showFamilyTree
+	 * @param {boolean} saleDescription
+	 * @param {boolean} applyLaw
+	 * @returns {DocumentFragment}
+	 */
+	function MultipleInspectDOM(slaves, showFamilyTree, saleDescription, applyLaw) {
+		const frag = document.createDocumentFragment();
+		const tabbar = App.UI.DOM.appendNewElement("div", "", frag, "tabbar");
+
+		for (const slave of slaves) {
+			tabbar.append(App.UI.tabbar.tabButtonDOM(`slave${slave.ID}`, slave.slaveName));
+			frag.append(App.UI.tabbar.makeTabDOM(`slave${slave.ID}`, _LSDDOM(slave, saleDescription, applyLaw)));
+		}
+
+		if (slaves.length > 1 && showFamilyTree) {
+			const button = App.UI.tabbar.tabButtonDOM(`familyTreeTab`, "Family Tree");
+			button.addEventListener('click', event => {
+				renderFamilyTree(slaves, slaves[0].ID);
+			});
+			tabbar.append(button);
+			const ftTarget = document.createElement("div");
+			ftTarget.setAttribute("id", "familyTree");
+			frag.append(App.UI.tabbar.makeTabDOM(`familyTreeTab`, ftTarget));
+		}
+
+		App.UI.tabbar.handlePreSelectedTab(`slave${slaves[0].ID}`);
+
+		return frag;
+	}
+
+	/**
+	 * Provide a mechanism to inspect multiple slaves at once (for example, for Household Liquidators and recETS).
+	 * Usable directly from SugarCube passages.
+	 * @param {Array<App.Entity.SlaveState>} slaves
+	 * @param {boolean} showFamilyTree
+	 * @param {boolean} saleDescription
+	 * @param {boolean} applyLaw
+	 * @returns {string}
+	 */
+	function MultipleInspectSugarcube(slaves, showFamilyTree, saleDescription, applyLaw) {
+		const frag = MultipleInspectDOM(slaves, showFamilyTree, saleDescription, applyLaw);
+		return App.UI.DOM.includeDOM(frag, "MultipleInspect", "div");
+	}
+
+	return {
+		DOM: MultipleInspectDOM,
+		SC: MultipleInspectSugarcube
+	};
+})();
diff --git a/src/uncategorized/householdLiquidator.tw b/src/uncategorized/householdLiquidator.tw
index 967f065dbbc83546e1c13643edf9c02b32b4edf0..4fbb82fe55ffeab928139598cf75a2f62165dae6 100644
--- a/src/uncategorized/householdLiquidator.tw
+++ b/src/uncategorized/householdLiquidator.tw
@@ -23,7 +23,7 @@
 	<<set _oppositeSex = 1>>
 <</if>>
 
-The household liquidator is offering a set of siblings for sale. As usual, you will only be permitted to inspect the older, but there is a guarantee that the younger will be similar.
+The household liquidator is offering a set of siblings for sale. You are permitted to inspect both slaves.
 <br><br>
 
 <<set _relativeSlave = generateRelatedSlave($activeSlave, "younger sibling", _oppositeSex)>>
@@ -61,7 +61,7 @@ The household liquidator is offering a set of siblings for sale. As usual, you w
 <</if>>
 
 <<setLocalPronouns $activeSlave>>
-The household liquidator is offering a $mother and $his <<if $activeSlave.genes == "XX" && _oppositeSex == 1>>son<<elseif $activeSlave.genes == "XY" && _oppositeSex == 1>>daughter<<else>>$daughter<</if>> for sale. As usual, you will only be permitted to inspect the $mother, but there is a guarantee that the <<if $activeSlave.genes == "XX" && _oppositeSex == 1>>son<<elseif $activeSlave.genes == "XY" && _oppositeSex == 1>>daughter<<else>>$daughter<</if>> will be similar.
+The household liquidator is offering a $mother and $his <<if $activeSlave.genes == "XX" && _oppositeSex == 1>>son<<elseif $activeSlave.genes == "XY" && _oppositeSex == 1>>daughter<<else>>$daughter<</if>> for sale. You are permitted to inspect both slaves.
 <br><br>
 
 <<set _relativeSlave = generateRelatedSlave($activeSlave, "child", _oppositeSex)>>
@@ -107,5 +107,4 @@ The price is <<print cashFormatColor(_totalCost)>>. <<if $slavesSeen > $slaveMar
 <br>[[Decline to purchase them and check out another set of slaves|Household Liquidator][$slavesSeen += 2]]
 <br><br>
 
-<<set $saleDescription = 1, $applyLaw = 1>>
-<<include "Long Slave Description">>
+<<= App.UI.MultipleInspect.SC(_newSlaves, true, true, true)>>
diff --git a/src/uncategorized/reFSEgyptianRevivalistAcquisition.tw b/src/uncategorized/reFSEgyptianRevivalistAcquisition.tw
index 5d2b3634a7f71bf6cb21a733d583fc446ca2cc0d..8a8a11697fe3d41851741686070c0f7a9c2a0fb7 100644
--- a/src/uncategorized/reFSEgyptianRevivalistAcquisition.tw
+++ b/src/uncategorized/reFSEgyptianRevivalistAcquisition.tw
@@ -62,12 +62,13 @@ You receive so many messages, as a noted titan of the new Free Cities world, tha
 <br><br>
 This call is coming from a public kiosk, which is usually an indication that the person on the other end is a transient individual who has decided to take slavery over homelessness. In this case, however, the story is more unusual — the callers seem stressed, but otherwise normal. They haltingly and quietly explain, with many nervous glances off-camera to ensure they are not overheard, that they are both siblings and lovers, and their attempts to keep the truth of the nature of their relationship from their friends, family, and society at large have failed. They had heard of $arcologies[0].name's reverence for incestuous relationships, and have managed to talk themselves into the questionable conclusion that their only chance to be together was for them to sell themselves to someone who would not just accept but encourage their incest — namely, you.
 <br><br>
+<<= capFirstChar($assistant.name)>> assembles a dossier of information and photos from information they've sent describing their bodies and skills, to be used as a substitute for an in-person inspection.
+<br></br>
 
 <<set _totalValue = slaveCost($activeSlave) + slaveCost(_secondSlave)>>
 //Enslaving them will cost <<print cashFormat(_contractCost)>>. Selling them immediately will bring in approximately <<print cashFormat(_totalValue-_contractCost)>>.//
 <br><br>
-<<set $saleDescription = 0>>
-<<include "Long Slave Description">>
+<<= App.UI.MultipleInspect.SC(_newSlaves, true, false, false)>>
 <br><br>
 <span id="result">
 <<if $cash >= _contractCost>>
diff --git a/src/uncategorized/recETS.tw b/src/uncategorized/recETS.tw
index dad4abeab5d5357722cd34904eead199906af106..01edf36290279ea832f23ae4e4fc87dc252bb836 100644
--- a/src/uncategorized/recETS.tw
+++ b/src/uncategorized/recETS.tw
@@ -1365,6 +1365,13 @@
 				"We co<<s>>t <<print cashFormat(_ContractCost)>>, <<if $PC.title != 0>><<s>>ir<<else>>ma'am<</if>>."
 		<</switch>>
 
+		/* Inspection details for videocall recruitment */
+		<<if ["incest mother son", "incest father daughter", "incest brother sister", "incest sister sister", "incest brother brother", "incest twin sister", "incest twin brother", "incest twins mixed", "incest mother daughter", "incest father son"].includes($RecETSevent)>>
+			<p>
+				<<= capFirstChar($assistant.name)>> assembles a dossier of information and photos from information they've sent describing their bodies and skills, to be used as a substitute for an in-person inspection.
+			</p>
+		<</if>>
+
 		/* The price */
 		<p class="detail">
 			<<switch $RecETSevent>>
@@ -1383,7 +1390,7 @@
 				<<case "identical herm pair">>
 					$His _sister2 is identical.
 				<<case "desperate broodmother">>
-					Enslaving $him will cost <<print cashFormat(_ContractCost)>>.<<if $minimumSlaveAge <= 3>> Alternatively, for <<print cashFormat((_ContractCost*2))>> you could enslave both mother and $his eldest child, the rest being sent to the slave orphanages, but you can't tell its gender from the call. You can expect a good value, though.<</if>>
+					Enslaving $him will cost <<print cashFormat(_ContractCost)>>.<<if $minimumSlaveAge <= 3>> Alternatively, for <<print cashFormat((_ContractCost*2))>> you could enslave both mother and $his eldest child, the rest being sent to the slave orphanages.<</if>>
 				<<default>>
 					<div class="error" font-style="normal">
 						ERROR: bad recETS event $RecETSevent
@@ -1391,10 +1398,13 @@
 			<</switch>>
 		</p>
 
-
-		<p>
-			<<set $saleDescription = 1>><<include "Long Slave Description">><<set $saleDescription = 0>>
-		</p>
+		<<set _newSlaves = [$relative, $relative2]>>
+		<<if $RecETSevent === "desperate broodmother">>
+			<<set _newSlaves = [$relative]>> /* rushed videocall, no dossier */
+		<<elseif ["addict mother daughter", "posh mother daughter", "mismatched pair"].includes($RecETSevent)>>
+			<<set _newSlaves = [$relative]>> /* caller doesn't want relative involved, so you don't get to inspect her even if you can force a sale */
+		<</if>>
+		<<= App.UI.MultipleInspect.SC(_newSlaves, true, true, false)>>
 	</span>
 
 	<span id="result">