xref: /haiku/src/servers/package/CommitTransactionHandler.cpp (revision 47c05920fde47c2618efccd24bd82f1e79cdf05a)
1 /*
2  * Copyright 2013-2014, Haiku, Inc. All Rights Reserved.
3  * Distributed under the terms of the MIT License.
4  *
5  * Authors:
6  *		Ingo Weinhold <ingo_weinhold@gmx.de>
7  */
8 
9 
10 #include "CommitTransactionHandler.h"
11 
12 #include <errno.h>
13 #include <grp.h>
14 #include <pwd.h>
15 
16 #include <File.h>
17 #include <Path.h>
18 #include <SymLink.h>
19 
20 #include <AutoDeleter.h>
21 #include <CopyEngine.h>
22 #include <NotOwningEntryRef.h>
23 #include <package/CommitTransactionResult.h>
24 #include <package/DaemonDefs.h>
25 #include <RemoveEngine.h>
26 
27 #include "Constants.h"
28 #include "DebugSupport.h"
29 #include "Exception.h"
30 #include "PackageFileManager.h"
31 #include "VolumeState.h"
32 
33 
34 using namespace BPackageKit::BPrivate;
35 
36 using BPackageKit::BTransactionIssue;
37 
38 
39 // #pragma mark - TransactionIssueBuilder
40 
41 
42 struct CommitTransactionHandler::TransactionIssueBuilder {
43 	TransactionIssueBuilder(BTransactionIssue::BType type,
44 		Package* package = NULL)
45 		:
46 		fType(type),
47 		fPackageName(package != NULL ? package->FileName() : BString()),
48 		fPath1(),
49 		fPath2(),
50 		fSystemError(B_OK),
51 		fExitCode(0)
52 	{
53 	}
54 
55 	TransactionIssueBuilder& SetPath1(const BString& path)
56 	{
57 		fPath1 = path;
58 		return *this;
59 	}
60 
61 	TransactionIssueBuilder& SetPath1(const FSUtils::Entry& entry)
62 	{
63 		return SetPath1(entry.Path());
64 	}
65 
66 	TransactionIssueBuilder& SetPath2(const BString& path)
67 	{
68 		fPath2 = path;
69 		return *this;
70 	}
71 
72 	TransactionIssueBuilder& SetPath2(const FSUtils::Entry& entry)
73 	{
74 		return SetPath2(entry.Path());
75 	}
76 
77 	TransactionIssueBuilder& SetSystemError(status_t error)
78 	{
79 		fSystemError = error;
80 		return *this;
81 	}
82 
83 	TransactionIssueBuilder& SetExitCode(int exitCode)
84 	{
85 		fExitCode = exitCode;
86 		return *this;
87 	}
88 
89 	BTransactionIssue BuildIssue(Package* package) const
90 	{
91 		BString packageName(fPackageName);
92 		if (packageName.IsEmpty() && package != NULL)
93 			packageName = package->FileName();
94 
95 		return BTransactionIssue(fType, packageName, fPath1, fPath2,
96 			fSystemError, fExitCode);
97 	}
98 
99 private:
100 	BTransactionIssue::BType	fType;
101 	BString						fPackageName;
102 	BString						fPath1;
103 	BString						fPath2;
104 	status_t					fSystemError;
105 	int							fExitCode;
106 };
107 
108 
109 // #pragma mark - CommitTransactionHandler
110 
111 
112 CommitTransactionHandler::CommitTransactionHandler(Volume* volume,
113 	PackageFileManager* packageFileManager, BCommitTransactionResult& result)
114 	:
115 	fVolume(volume),
116 	fPackageFileManager(packageFileManager),
117 	fVolumeState(NULL),
118 	fVolumeStateIsActive(false),
119 	fPackagesToActivate(),
120 	fPackagesToDeactivate(),
121 	fAddedPackages(),
122 	fRemovedPackages(),
123 	fPackagesAlreadyAdded(),
124 	fPackagesAlreadyRemoved(),
125 	fOldStateDirectory(),
126 	fOldStateDirectoryRef(),
127 	fOldStateDirectoryName(),
128 	fTransactionDirectoryRef(),
129 	fFirstBootProcessing(false),
130 	fWritableFilesDirectory(),
131 	fAddedGroups(),
132 	fAddedUsers(),
133 	fFSTransaction(),
134 	fResult(result),
135 	fCurrentPackage(NULL)
136 {
137 }
138 
139 
140 CommitTransactionHandler::~CommitTransactionHandler()
141 {
142 	// Delete Package objects we created in case of error (on success
143 	// fPackagesToActivate will be empty).
144 	int32 count = fPackagesToActivate.CountItems();
145 	for (int32 i = 0; i < count; i++) {
146 		Package* package = fPackagesToActivate.ItemAt(i);
147 		if (fPackagesAlreadyAdded.find(package)
148 				== fPackagesAlreadyAdded.end()) {
149 			delete package;
150 		}
151 	}
152 
153 	delete fVolumeState;
154 }
155 
156 
157 void
158 CommitTransactionHandler::Init(VolumeState* volumeState,
159 	bool isActiveVolumeState, const PackageSet& packagesAlreadyAdded,
160 	const PackageSet& packagesAlreadyRemoved)
161 {
162 	fVolumeState = volumeState->Clone();
163 	if (fVolumeState == NULL)
164 		throw std::bad_alloc();
165 
166 	fVolumeStateIsActive = isActiveVolumeState;
167 
168 	for (PackageSet::const_iterator it = packagesAlreadyAdded.begin();
169 			it != packagesAlreadyAdded.end(); ++it) {
170 		Package* package = fVolumeState->FindPackage((*it)->FileName());
171 		fPackagesAlreadyAdded.insert(package);
172 	}
173 
174 	for (PackageSet::const_iterator it = packagesAlreadyRemoved.begin();
175 			it != packagesAlreadyRemoved.end(); ++it) {
176 		Package* package = fVolumeState->FindPackage((*it)->FileName());
177 		fPackagesAlreadyRemoved.insert(package);
178 	}
179 }
180 
181 
182 void
183 CommitTransactionHandler::HandleRequest(BMessage* request)
184 {
185 	status_t error;
186 
187 	BActivationTransaction transaction(request, &error);
188 	if (error == B_OK)
189 		error = transaction.InitCheck();
190 	if (error != B_OK) {
191 		if (error == B_NO_MEMORY)
192 			throw Exception(B_TRANSACTION_NO_MEMORY);
193 		throw Exception(B_TRANSACTION_BAD_REQUEST);
194 	}
195 
196 	HandleRequest(transaction);
197 }
198 
199 
200 void
201 CommitTransactionHandler::HandleRequest(
202 	const BActivationTransaction& transaction)
203 {
204 	// check the change count
205 	if (transaction.ChangeCount() != fVolume->ChangeCount())
206 		throw Exception(B_TRANSACTION_CHANGE_COUNT_MISMATCH);
207 
208 	fFirstBootProcessing = transaction.FirstBootProcessing();
209 
210 	// collect the packages to deactivate
211 	_GetPackagesToDeactivate(transaction);
212 
213 	// read the packages to activate
214 	_ReadPackagesToActivate(transaction);
215 
216 	// anything to do at all?
217 	if (fPackagesToActivate.IsEmpty() && fPackagesToDeactivate.empty()) {
218 		WARN("Bad package activation request: no packages to activate or"
219 			" deactivate\n");
220 		throw Exception(B_TRANSACTION_BAD_REQUEST);
221 	}
222 
223 	_ApplyChanges();
224 
225 	// Clean up the unused empty transaction directory for first boot
226 	// processing, since it's usually an internal to package_daemon
227 	// operation and there is no external client to clean it up.
228 	if (fFirstBootProcessing) {
229 		RelativePath directoryPath(kAdminDirectoryName,
230 			transaction.TransactionDirectoryName().String());
231 		BDirectory transactionDir;
232 		status_t error = _OpenPackagesSubDirectory(directoryPath, false,
233 			transactionDir);
234 		if (error == B_OK) {
235 			BEntry transactionDirEntry;
236 			error = transactionDir.GetEntry(&transactionDirEntry);
237 			if (error == B_OK)
238 				transactionDirEntry.Remove(); // Okay to fail when non-empty.
239 		}
240 	}
241 }
242 
243 
244 void
245 CommitTransactionHandler::HandleRequest()
246 {
247 	for (PackageSet::const_iterator it = fPackagesAlreadyAdded.begin();
248 		it != fPackagesAlreadyAdded.end(); ++it) {
249 		if (!fPackagesToActivate.AddItem(*it))
250 			throw std::bad_alloc();
251 	}
252 
253 	fPackagesToDeactivate = fPackagesAlreadyRemoved;
254 
255 	_ApplyChanges();
256 }
257 
258 
259 void
260 CommitTransactionHandler::Revert()
261 {
262 	// move packages to activate back to transaction directory
263 	_RevertAddPackagesToActivate();
264 
265 	// move packages to deactivate back to packages directory
266 	_RevertRemovePackagesToDeactivate();
267 
268 	// revert user and group changes
269 	_RevertUserGroupChanges();
270 
271 	// Revert all other FS operations, i.e. the writable files changes as
272 	// well as the creation of the old state directory.
273 	fFSTransaction.RollBack();
274 }
275 
276 
277 VolumeState*
278 CommitTransactionHandler::DetachVolumeState()
279 {
280 	VolumeState* result = fVolumeState;
281 	fVolumeState = NULL;
282 	return result;
283 }
284 
285 
286 void
287 CommitTransactionHandler::_GetPackagesToDeactivate(
288 	const BActivationTransaction& transaction)
289 {
290 	// get the number of packages to deactivate
291 	const BStringList& packagesToDeactivate
292 		= transaction.PackagesToDeactivate();
293 	int32 packagesToDeactivateCount = packagesToDeactivate.CountStrings();
294 	if (packagesToDeactivateCount == 0)
295 		return;
296 
297 	for (int32 i = 0; i < packagesToDeactivateCount; i++) {
298 		BString packageName = packagesToDeactivate.StringAt(i);
299 		Package* package = fVolumeState->FindPackage(packageName);
300 		if (package == NULL) {
301 			throw Exception(B_TRANSACTION_NO_SUCH_PACKAGE)
302 				.SetPackageName(packageName);
303 		}
304 
305 		fPackagesToDeactivate.insert(package);
306 	}
307 }
308 
309 
310 void
311 CommitTransactionHandler::_ReadPackagesToActivate(
312 	const BActivationTransaction& transaction)
313 {
314 	// get the number of packages to activate
315 	const BStringList& packagesToActivate
316 		= transaction.PackagesToActivate();
317 	int32 packagesToActivateCount = packagesToActivate.CountStrings();
318 	if (packagesToActivateCount == 0)
319 		return;
320 
321 	// check the transaction directory name -- we only allow a simple
322 	// subdirectory of the admin directory
323 	const BString& transactionDirectoryName
324 		= transaction.TransactionDirectoryName();
325 	if (transactionDirectoryName.IsEmpty()
326 		|| transactionDirectoryName.FindFirst('/') >= 0
327 		|| transactionDirectoryName == "."
328 		|| transactionDirectoryName == "..") {
329 		WARN("Bad package activation request: malformed transaction"
330 			" directory name: \"%s\"\n", transactionDirectoryName.String());
331 		throw Exception(B_TRANSACTION_BAD_REQUEST);
332 	}
333 
334 	// open the directory
335 	RelativePath directoryPath(kAdminDirectoryName,
336 		transactionDirectoryName);
337 	BDirectory directory;
338 	status_t error = _OpenPackagesSubDirectory(directoryPath, false, directory);
339 	if (error == B_OK) {
340 		error = directory.GetNodeRef(&fTransactionDirectoryRef);
341 		if (error != B_OK) {
342 			ERROR("Failed to get transaction directory node ref: %s\n",
343 				strerror(error));
344 		}
345 	} else
346 		ERROR("Failed to open transaction directory: %s\n", strerror(error));
347 
348 	if (error != B_OK) {
349 		throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
350 			.SetPath1(_GetPath(
351 				FSUtils::Entry(fVolume->PackagesDirectoryRef(),
352 					directoryPath.ToString()),
353 				directoryPath.ToString()))
354 			.SetSystemError(error);
355 	}
356 
357 	// read the packages
358 	for (int32 i = 0; i < packagesToActivateCount; i++) {
359 		BString packageName = packagesToActivate.StringAt(i);
360 		// make sure it doesn't clash with an already existing package,
361 		// except in first boot mode where it should always clash.
362 		Package* package = fVolumeState->FindPackage(packageName);
363 		if (fFirstBootProcessing) {
364 			if (package == NULL) {
365 				throw Exception(B_TRANSACTION_NO_SUCH_PACKAGE)
366 					.SetPackageName(packageName);
367 			}
368 			if (!fPackagesToActivate.AddItem(package))
369 				throw Exception(B_TRANSACTION_NO_MEMORY);
370 			continue;
371 		} else {
372 			if (package != NULL) {
373 				if (fPackagesAlreadyAdded.find(package)
374 						!= fPackagesAlreadyAdded.end()) {
375 					if (!fPackagesToActivate.AddItem(package))
376 						throw Exception(B_TRANSACTION_NO_MEMORY);
377 					continue;
378 				}
379 
380 				if (fPackagesToDeactivate.find(package)
381 						== fPackagesToDeactivate.end()) {
382 					throw Exception(B_TRANSACTION_PACKAGE_ALREADY_EXISTS)
383 						.SetPackageName(packageName);
384 				}
385 			}
386 		}
387 
388 		// read the package
389 		error = fPackageFileManager->CreatePackage(
390 			NotOwningEntryRef(fTransactionDirectoryRef, packageName),
391 			package);
392 		if (error != B_OK) {
393 			if (error == B_NO_MEMORY)
394 				throw Exception(B_TRANSACTION_NO_MEMORY);
395 			throw Exception(B_TRANSACTION_FAILED_TO_READ_PACKAGE_FILE)
396 				.SetPackageName(packageName)
397 				.SetPath1(_GetPath(
398 					FSUtils::Entry(
399 						NotOwningEntryRef(fTransactionDirectoryRef,
400 							packageName)),
401 					packageName))
402 				.SetSystemError(error);
403 		}
404 
405 		if (!fPackagesToActivate.AddItem(package)) {
406 			delete package;
407 			throw Exception(B_TRANSACTION_NO_MEMORY);
408 		}
409 	}
410 }
411 
412 
413 void
414 CommitTransactionHandler::_ApplyChanges()
415 {
416 	if (!fFirstBootProcessing)
417 	{
418 		// create an old state directory
419 		_CreateOldStateDirectory();
420 
421 		// move packages to deactivate to old state directory
422 		_RemovePackagesToDeactivate();
423 
424 		// move packages to activate to packages directory
425 		_AddPackagesToActivate();
426 
427 		// run pre-uninstall scripts, before their packages vanish.
428 		_RunPreUninstallScripts();
429 
430 		// activate/deactivate packages and create users, groups, settings files.
431 		_ChangePackageActivation(fAddedPackages, fRemovedPackages);
432 	} else // FirstBootProcessing, skip several steps and just do package setup.
433 		_PrepareFirstBootPackages();
434 
435 	// run post-install scripts now that the new packages are visible in the
436 	// package file system.
437 	if (fVolumeStateIsActive || fFirstBootProcessing) {
438 		_RunPostInstallScripts();
439 	} else {
440 		// Do post-install scripts later after a reboot, for Haiku OS packages.
441 		_QueuePostInstallScripts();
442 	}
443 
444 	// removed packages have been deleted, new packages shall not be deleted
445 	fAddedPackages.clear();
446 	fRemovedPackages.clear();
447 	fPackagesToActivate.MakeEmpty(false);
448 	fPackagesToDeactivate.clear();
449 }
450 
451 
452 void
453 CommitTransactionHandler::_CreateOldStateDirectory()
454 {
455 	time_t stateTime = 0;
456 	{
457 		// use the modification time of the old activations file, if possible
458 		BFile oldActivationFile;
459 		BEntry oldActivationEntry;
460 		if (_OpenPackagesFile(RelativePath(kAdminDirectoryName), kActivationFileName,
461 				B_READ_ONLY, oldActivationFile, &oldActivationEntry) != B_OK
462 					|| oldActivationEntry.GetModificationTime(&stateTime) != B_OK) {
463 			stateTime = time(NULL);
464 		}
465 	}
466 
467 	// construct a nice name from the date and time
468 	struct tm now;
469 	BString baseName;
470 	if (localtime_r(&stateTime, &now) != NULL) {
471 		baseName.SetToFormat("state_%d-%02d-%02d_%02d:%02d:%02d",
472 			1900 + now.tm_year, now.tm_mon + 1, now.tm_mday, now.tm_hour,
473 			now.tm_min, now.tm_sec);
474 	} else
475 		baseName = "state";
476 
477 	if (baseName.IsEmpty())
478 		throw Exception(B_TRANSACTION_NO_MEMORY);
479 
480 	// make sure the directory doesn't exist yet
481 	BDirectory adminDirectory;
482 	status_t error = _OpenPackagesSubDirectory(
483 		RelativePath(kAdminDirectoryName), true, adminDirectory);
484 	if (error != B_OK) {
485 		ERROR("Failed to open administrative directory: %s\n", strerror(error));
486 		throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
487 			.SetPath1(_GetPath(
488 				FSUtils::Entry(fVolume->PackagesDirectoryRef(),
489 					kAdminDirectoryName),
490 				kAdminDirectoryName))
491 			.SetSystemError(error);
492 	}
493 
494 	int uniqueId = 1;
495 	BString directoryName = baseName;
496 	while (BEntry(&adminDirectory, directoryName).Exists()) {
497 		directoryName.SetToFormat("%s-%d", baseName.String(), uniqueId++);
498 		if (directoryName.IsEmpty())
499 			throw Exception(B_TRANSACTION_NO_MEMORY);
500 	}
501 
502 	// create the directory
503 	FSTransaction::CreateOperation createOldStateDirectoryOperation(
504 		&fFSTransaction, FSUtils::Entry(adminDirectory, directoryName));
505 
506 	error = adminDirectory.CreateDirectory(directoryName,
507 		&fOldStateDirectory);
508 	if (error == B_OK) {
509 		createOldStateDirectoryOperation.Finished();
510 
511 		fOldStateDirectoryName = directoryName;
512 
513 		error = fOldStateDirectory.GetNodeRef(&fOldStateDirectoryRef);
514 		if (error != B_OK)
515 			ERROR("Failed get old state directory ref: %s\n", strerror(error));
516 	} else
517 		ERROR("Failed to create old state directory: %s\n", strerror(error));
518 
519 	if (error != B_OK) {
520 		throw Exception(B_TRANSACTION_FAILED_TO_CREATE_DIRECTORY)
521 			.SetPath1(_GetPath(
522 				FSUtils::Entry(adminDirectory, directoryName),
523 				directoryName))
524 			.SetSystemError(error);
525 	}
526 
527 	// write the old activation file
528 	BEntry activationFile;
529 	_WriteActivationFile(RelativePath(kAdminDirectoryName, directoryName),
530 		kActivationFileName, PackageSet(), PackageSet(), activationFile);
531 
532 	fResult.SetOldStateDirectory(fOldStateDirectoryName);
533 }
534 
535 
536 void
537 CommitTransactionHandler::_RemovePackagesToDeactivate()
538 {
539 	if (fPackagesToDeactivate.empty())
540 		return;
541 
542 	for (PackageSet::const_iterator it = fPackagesToDeactivate.begin();
543 		it != fPackagesToDeactivate.end(); ++it) {
544 		Package* package = *it;
545 
546 		// When deactivating (or updating) a system package, don't do that live.
547 		if (_IsSystemPackage(package))
548 			fVolumeStateIsActive = false;
549 
550 		if (fPackagesAlreadyRemoved.find(package)
551 				!= fPackagesAlreadyRemoved.end()) {
552 			fRemovedPackages.insert(package);
553 			continue;
554 		}
555 
556 		// get a BEntry for the package
557 		NotOwningEntryRef entryRef(package->EntryRef());
558 
559 		BEntry entry;
560 		status_t error = entry.SetTo(&entryRef);
561 		if (error != B_OK) {
562 			ERROR("Failed to get package entry for %s: %s\n",
563 				package->FileName().String(), strerror(error));
564 			throw Exception(B_TRANSACTION_FAILED_TO_GET_ENTRY_PATH)
565 				.SetPath1(package->FileName())
566 				.SetPackageName(package->FileName())
567 				.SetSystemError(error);
568 		}
569 
570 		// move entry
571 		fRemovedPackages.insert(package);
572 
573 		error = entry.MoveTo(&fOldStateDirectory);
574 		if (error != B_OK) {
575 			fRemovedPackages.erase(package);
576 			ERROR("Failed to move old package %s from packages directory: %s\n",
577 				package->FileName().String(), strerror(error));
578 			throw Exception(B_TRANSACTION_FAILED_TO_MOVE_FILE)
579 				.SetPath1(
580 					_GetPath(FSUtils::Entry(entryRef), package->FileName()))
581 				.SetPath2(_GetPath(
582 					FSUtils::Entry(fOldStateDirectory),
583 					fOldStateDirectoryName))
584 				.SetSystemError(error);
585 		}
586 
587 		fPackageFileManager->PackageFileMoved(package->File(),
588 			fOldStateDirectoryRef);
589 		package->File()->IncrementEntryRemovedIgnoreLevel();
590 	}
591 }
592 
593 
594 void
595 CommitTransactionHandler::_AddPackagesToActivate()
596 {
597 	if (fPackagesToActivate.IsEmpty())
598 		return;
599 
600 	// open packages directory
601 	BDirectory packagesDirectory;
602 	status_t error
603 		= packagesDirectory.SetTo(&fVolume->PackagesDirectoryRef());
604 	if (error != B_OK) {
605 		ERROR("Failed to open packages directory: %s\n", strerror(error));
606 		throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
607 			.SetPath1("<packages>")
608 			.SetSystemError(error);
609 	}
610 
611 	int32 count = fPackagesToActivate.CountItems();
612 	for (int32 i = 0; i < count; i++) {
613 		Package* package = fPackagesToActivate.ItemAt(i);
614 		if (fPackagesAlreadyAdded.find(package)
615 				!= fPackagesAlreadyAdded.end()) {
616 			fAddedPackages.insert(package);
617 			_PreparePackageToActivate(package);
618 			continue;
619 		}
620 
621 		// get a BEntry for the package
622 		NotOwningEntryRef entryRef(fTransactionDirectoryRef,
623 			package->FileName());
624 		BEntry entry;
625 		error = entry.SetTo(&entryRef);
626 		if (error != B_OK) {
627 			ERROR("Failed to get package entry for %s: %s\n",
628 				package->FileName().String(), strerror(error));
629 			throw Exception(B_TRANSACTION_FAILED_TO_GET_ENTRY_PATH)
630 				.SetPath1(package->FileName())
631 				.SetPackageName(package->FileName())
632 				.SetSystemError(error);
633 		}
634 
635 		// move entry
636 		fAddedPackages.insert(package);
637 
638 		error = entry.MoveTo(&packagesDirectory);
639 		if (error == B_FILE_EXISTS) {
640 			error = _AssertEntriesAreEqual(entry, &packagesDirectory);
641 			if (error == B_OK) {
642 				// Packages are identical, no need to move.
643 				// If the entry is not removed however, it will prevent
644 				// the transaction directory from being removed later.
645 				// We ignore failure to Remove() here, though.
646 				entry.Remove();
647 			} else if (error != B_FILE_EXISTS) {
648 				ERROR("Failed to compare new package %s to existing file in "
649 					"packages directory: %s\n", package->FileName().String(),
650 					strerror(error));
651 				// Restore original error to avoid confusion
652 				error = B_FILE_EXISTS;
653 			}
654 		}
655 		if (error != B_OK) {
656 			fAddedPackages.erase(package);
657 			ERROR("Failed to move new package %s to packages directory: %s\n",
658 				package->FileName().String(), strerror(error));
659 			throw Exception(B_TRANSACTION_FAILED_TO_MOVE_FILE)
660 				.SetPath1(
661 					_GetPath(FSUtils::Entry(entryRef), package->FileName()))
662 				.SetPath2(_GetPath(
663 					FSUtils::Entry(packagesDirectory),
664 					"packages"))
665 				.SetSystemError(error);
666 		}
667 
668 		fPackageFileManager->PackageFileMoved(package->File(),
669 			fVolume->PackagesDirectoryRef());
670 		package->File()->IncrementEntryCreatedIgnoreLevel();
671 
672 		// also add the package to the volume
673 		fVolumeState->AddPackage(package);
674 
675 		_PreparePackageToActivate(package);
676 	}
677 }
678 
679 
680 void
681 CommitTransactionHandler::_PrepareFirstBootPackages()
682 {
683 	int32 count = fPackagesToActivate.CountItems();
684 
685 	BDirectory transactionDir(&fTransactionDirectoryRef);
686 	BEntry transactionEntry;
687 	BPath transactionPath;
688 	if (transactionDir.InitCheck() == B_OK &&
689 			transactionDir.GetEntry(&transactionEntry) == B_OK &&
690 			transactionEntry.GetPath(&transactionPath) == B_OK) {
691 		INFORM("Starting First Boot Processing for %d packages in %s.\n",
692 			(int) count, transactionPath.Path());
693 	}
694 
695 	for (int32 i = 0; i < count; i++) {
696 		Package* package = fPackagesToActivate.ItemAt(i);
697 		fAddedPackages.insert(package);
698 		INFORM("Doing first boot processing #%d for package %s.\n",
699 			(int) i, package->FileName().String());
700 		_PreparePackageToActivate(package);
701 	}
702 }
703 
704 
705 void
706 CommitTransactionHandler::_PreparePackageToActivate(Package* package)
707 {
708 	fCurrentPackage = package;
709 
710 	// add groups
711 	const BStringList& groups = package->Info().Groups();
712 	int32 count = groups.CountStrings();
713 	for (int32 i = 0; i < count; i++)
714 		_AddGroup(package, groups.StringAt(i));
715 
716 	// add users
717 	const BObjectList<BUser>& users = package->Info().Users();
718 	for (int32 i = 0; const BUser* user = users.ItemAt(i); i++)
719 		_AddUser(package, *user);
720 
721 	// handle global writable files
722 	_AddGlobalWritableFiles(package);
723 
724 	fCurrentPackage = NULL;
725 }
726 
727 
728 void
729 CommitTransactionHandler::_AddGroup(Package* package, const BString& groupName)
730 {
731 	// Check whether the group already exists.
732 	char buffer[256];
733 	struct group groupBuffer;
734 	struct group* groupFound;
735 	int error = getgrnam_r(groupName, &groupBuffer, buffer, sizeof(buffer),
736 		&groupFound);
737 	if ((error == 0 && groupFound != NULL) || error == ERANGE)
738 		return;
739 
740 	// add it
741 	fAddedGroups.insert(groupName.String());
742 
743 	std::string commandLine("groupadd ");
744 	commandLine += FSUtils::ShellEscapeString(groupName).String();
745 
746 	if (system(commandLine.c_str()) != 0) {
747 		fAddedGroups.erase(groupName.String());
748 		ERROR("Failed to add group \"%s\".\n", groupName.String());
749 		throw Exception(B_TRANSACTION_FAILED_TO_ADD_GROUP)
750 			.SetPackageName(package->FileName())
751 			.SetString1(groupName);
752 	}
753 }
754 
755 
756 void
757 CommitTransactionHandler::_AddUser(Package* package, const BUser& user)
758 {
759 	// Check whether the user already exists.
760 	char buffer[256];
761 	struct passwd passwdBuffer;
762 	struct passwd* passwdFound;
763 	int error = getpwnam_r(user.Name(), &passwdBuffer, buffer,
764 		sizeof(buffer), &passwdFound);
765 	if ((error == 0 && passwdFound != NULL) || error == ERANGE)
766 		return;
767 
768 	// add it
769 	fAddedUsers.insert(user.Name().String());
770 
771 	std::string commandLine("useradd ");
772 
773 	if (!user.RealName().IsEmpty()) {
774 		commandLine += std::string("-n ")
775 			+ FSUtils::ShellEscapeString(user.RealName()).String() + " ";
776 	}
777 
778 	if (!user.Home().IsEmpty()) {
779 		commandLine += std::string("-d ")
780 			+ FSUtils::ShellEscapeString(user.Home()).String() + " ";
781 	}
782 
783 	if (!user.Shell().IsEmpty()) {
784 		commandLine += std::string("-s ")
785 			+ FSUtils::ShellEscapeString(user.Shell()).String() + " ";
786 	}
787 
788 	if (!user.Groups().IsEmpty()) {
789 		commandLine += std::string("-g ")
790 			+ FSUtils::ShellEscapeString(user.Groups().First()).String()
791 			+ " ";
792 	}
793 
794 	commandLine += FSUtils::ShellEscapeString(user.Name()).String();
795 
796 	if (system(commandLine.c_str()) != 0) {
797 		fAddedUsers.erase(user.Name().String());
798 		ERROR("Failed to add user \"%s\".\n", user.Name().String());
799 		throw Exception(B_TRANSACTION_FAILED_TO_ADD_USER)
800 			.SetPackageName(package->FileName())
801 			.SetString1(user.Name());
802 
803 	}
804 
805 	// add the supplementary groups
806 	int32 groupCount = user.Groups().CountStrings();
807 	for (int32 i = 1; i < groupCount; i++) {
808 		commandLine = std::string("groupmod -A ")
809 			+ FSUtils::ShellEscapeString(user.Name()).String()
810 			+ " "
811 			+ FSUtils::ShellEscapeString(user.Groups().StringAt(i))
812 				.String();
813 		if (system(commandLine.c_str()) != 0) {
814 			fAddedUsers.erase(user.Name().String());
815 			ERROR("Failed to add user \"%s\" to group \"%s\".\n",
816 				user.Name().String(), user.Groups().StringAt(i).String());
817 			throw Exception(B_TRANSACTION_FAILED_TO_ADD_USER_TO_GROUP)
818 				.SetPackageName(package->FileName())
819 				.SetString1(user.Name())
820 				.SetString2(user.Groups().StringAt(i));
821 		}
822 	}
823 }
824 
825 
826 void
827 CommitTransactionHandler::_AddGlobalWritableFiles(Package* package)
828 {
829 	// get the list of included files
830 	const BObjectList<BGlobalWritableFileInfo>& files
831 		= package->Info().GlobalWritableFileInfos();
832 	BStringList contentPaths;
833 	for (int32 i = 0; const BGlobalWritableFileInfo* file = files.ItemAt(i);
834 		i++) {
835 		if (file->IsIncluded() && !contentPaths.Add(file->Path()))
836 			throw std::bad_alloc();
837 	}
838 
839 	if (contentPaths.IsEmpty())
840 		return;
841 
842 	// Open the root directory of the installation location where we will
843 	// extract the files -- that's the volume's root directory.
844 	BDirectory rootDirectory;
845 	status_t error = rootDirectory.SetTo(&fVolume->RootDirectoryRef());
846 	if (error != B_OK) {
847 		throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
848 			.SetPath1(_GetPath(
849 				FSUtils::Entry(fVolume->RootDirectoryRef()),
850 				"<packagefs root>"))
851 			.SetSystemError(error);
852 	}
853 
854 	// Open writable-files directory in the administrative directory.
855 	if (fWritableFilesDirectory.InitCheck() != B_OK) {
856 		RelativePath directoryPath(kAdminDirectoryName,
857 			kWritableFilesDirectoryName);
858 		error = _OpenPackagesSubDirectory(directoryPath, true,
859 			fWritableFilesDirectory);
860 
861 		if (error != B_OK) {
862 			throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
863 				.SetPath1(_GetPath(
864 					FSUtils::Entry(fVolume->PackagesDirectoryRef(),
865 						directoryPath.ToString()),
866 					directoryPath.ToString()))
867 				.SetPackageName(package->FileName())
868 				.SetSystemError(error);
869 		}
870 	}
871 
872 	// extract files into a subdir of the writable-files directory
873 	BDirectory extractedFilesDirectory;
874 	_ExtractPackageContent(package, contentPaths,
875 		fWritableFilesDirectory, extractedFilesDirectory);
876 
877 	for (int32 i = 0; const BGlobalWritableFileInfo* file = files.ItemAt(i);
878 		i++) {
879 		if (file->IsIncluded()) {
880 			_AddGlobalWritableFile(package, *file, rootDirectory,
881 				extractedFilesDirectory);
882 		}
883 	}
884 }
885 
886 
887 void
888 CommitTransactionHandler::_AddGlobalWritableFile(Package* package,
889 	const BGlobalWritableFileInfo& file, const BDirectory& rootDirectory,
890 	const BDirectory& extractedFilesDirectory)
891 {
892 	// open parent directory of the source entry
893 	const char* lastSlash = strrchr(file.Path(), '/');
894 	const BDirectory* sourceDirectory;
895 	BDirectory stackSourceDirectory;
896 	if (lastSlash != NULL) {
897 		sourceDirectory = &stackSourceDirectory;
898 		BString sourceParentPath(file.Path(),
899 			lastSlash - file.Path().String());
900 		if (sourceParentPath.Length() == 0)
901 			throw std::bad_alloc();
902 
903 		status_t error = stackSourceDirectory.SetTo(
904 			&extractedFilesDirectory, sourceParentPath);
905 		if (error != B_OK) {
906 			throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
907 				.SetPath1(_GetPath(
908 					FSUtils::Entry(extractedFilesDirectory, sourceParentPath),
909 					sourceParentPath))
910 				.SetPackageName(package->FileName())
911 				.SetSystemError(error);
912 		}
913 	} else {
914 		sourceDirectory = &extractedFilesDirectory;
915 	}
916 
917 	// open parent directory of the target entry -- create, if necessary
918 	BString targetPath(file.Path());
919 	FSUtils::Path relativeSourcePath(file.Path());
920 	lastSlash = strrchr(targetPath, '/');
921 	if (lastSlash != NULL) {
922 		BString targetParentPath(targetPath,
923 			lastSlash - targetPath.String());
924 		if (targetParentPath.Length() == 0)
925 			throw std::bad_alloc();
926 
927 		BDirectory targetDirectory;
928 		status_t error = FSUtils::OpenSubDirectory(rootDirectory,
929 			RelativePath(targetParentPath), true, targetDirectory);
930 		if (error != B_OK) {
931 			throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
932 				.SetPath1(_GetPath(
933 					FSUtils::Entry(rootDirectory, targetParentPath),
934 					targetParentPath))
935 				.SetPackageName(package->FileName())
936 				.SetSystemError(error);
937 		}
938 		_AddGlobalWritableFileRecurse(package, *sourceDirectory,
939 			relativeSourcePath, targetDirectory, lastSlash + 1,
940 			file.UpdateType());
941 	} else {
942 		_AddGlobalWritableFileRecurse(package, *sourceDirectory,
943 			relativeSourcePath, rootDirectory, targetPath,
944 			file.UpdateType());
945 	}
946 }
947 
948 
949 void
950 CommitTransactionHandler::_AddGlobalWritableFileRecurse(Package* package,
951 	const BDirectory& sourceDirectory, FSUtils::Path& relativeSourcePath,
952 	const BDirectory& targetDirectory, const char* targetName,
953 	BWritableFileUpdateType updateType)
954 {
955 	// * If the file doesn't exist, just copy the extracted one.
956 	// * If the file does exist, compare with the previous original version:
957 	//   * If unchanged, just overwrite it.
958 	//   * If changed, leave it to the user for now. When we support merging
959 	//     first back the file up, then try the merge.
960 
961 	// Check whether the target location exists and what type the entry at
962 	// both locations are.
963 	struct stat targetStat;
964 	if (targetDirectory.GetStatFor(targetName, &targetStat) != B_OK) {
965 		// target doesn't exist -- just copy
966 		PRINT("Volume::CommitTransactionHandler::_AddGlobalWritableFile(): "
967 			"couldn't get stat for writable file \"%s\", copying...\n",
968 			targetName);
969 		FSTransaction::CreateOperation copyOperation(&fFSTransaction,
970 			FSUtils::Entry(targetDirectory, targetName));
971 		status_t error = BCopyEngine(BCopyEngine::COPY_RECURSIVELY)
972 			.CopyEntry(
973 				FSUtils::Entry(sourceDirectory, relativeSourcePath.Leaf()),
974 				FSUtils::Entry(targetDirectory, targetName));
975 		if (error != B_OK) {
976 			if (targetDirectory.GetStatFor(targetName, &targetStat) == B_OK)
977 				copyOperation.Finished();
978 
979 			throw Exception(B_TRANSACTION_FAILED_TO_COPY_FILE)
980 				.SetPath1(_GetPath(
981 					FSUtils::Entry(sourceDirectory,
982 						relativeSourcePath.Leaf()),
983 					relativeSourcePath))
984 				.SetPath2(_GetPath(
985 					FSUtils::Entry(targetDirectory, targetName),
986 					targetName))
987 				.SetSystemError(error);
988 		}
989 		copyOperation.Finished();
990 		return;
991 	}
992 
993 	struct stat sourceStat;
994 	status_t error = sourceDirectory.GetStatFor(relativeSourcePath.Leaf(),
995 		&sourceStat);
996 	if (error != B_OK) {
997 		throw Exception(B_TRANSACTION_FAILED_TO_ACCESS_ENTRY)
998 			.SetPath1(_GetPath(
999 				FSUtils::Entry(sourceDirectory,
1000 					relativeSourcePath.Leaf()),
1001 				relativeSourcePath))
1002 			.SetSystemError(error);
1003 	}
1004 
1005 	if ((sourceStat.st_mode & S_IFMT) != (targetStat.st_mode & S_IFMT)
1006 		|| (!S_ISDIR(sourceStat.st_mode) && !S_ISREG(sourceStat.st_mode)
1007 			&& !S_ISLNK(sourceStat.st_mode))) {
1008 		// Source and target entry types don't match or this is an entry
1009 		// we cannot handle. The user must handle this manually.
1010 		PRINT("Volume::CommitTransactionHandler::_AddGlobalWritableFile(): "
1011 			"writable file \"%s\" exists, but type doesn't match previous "
1012 			"type\n", targetName);
1013 		_AddIssue(TransactionIssueBuilder(
1014 				BTransactionIssue::B_WRITABLE_FILE_TYPE_MISMATCH)
1015 			.SetPath1(FSUtils::Entry(targetDirectory, targetName))
1016 			.SetPath2(FSUtils::Entry(sourceDirectory,
1017 				relativeSourcePath.Leaf())));
1018 		return;
1019 	}
1020 
1021 	if (S_ISDIR(sourceStat.st_mode)) {
1022 		// entry is a directory -- recurse
1023 		BDirectory sourceSubDirectory;
1024 		error = sourceSubDirectory.SetTo(&sourceDirectory,
1025 			relativeSourcePath.Leaf());
1026 		if (error != B_OK) {
1027 			throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
1028 				.SetPath1(_GetPath(
1029 					FSUtils::Entry(sourceDirectory,
1030 						relativeSourcePath.Leaf()),
1031 					relativeSourcePath))
1032 				.SetPackageName(package->FileName())
1033 				.SetSystemError(error);
1034 		}
1035 
1036 		BDirectory targetSubDirectory;
1037 		error = targetSubDirectory.SetTo(&targetDirectory, targetName);
1038 		if (error != B_OK) {
1039 			throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
1040 				.SetPath1(_GetPath(
1041 					FSUtils::Entry(targetDirectory, targetName),
1042 					targetName))
1043 				.SetPackageName(package->FileName())
1044 				.SetSystemError(error);
1045 		}
1046 
1047 		entry_ref entry;
1048 		while (sourceSubDirectory.GetNextRef(&entry) == B_OK) {
1049 			relativeSourcePath.AppendComponent(entry.name);
1050 			_AddGlobalWritableFileRecurse(package, sourceSubDirectory,
1051 				relativeSourcePath, targetSubDirectory, entry.name,
1052 				updateType);
1053 			relativeSourcePath.RemoveLastComponent();
1054 		}
1055 
1056 		PRINT("Volume::CommitTransactionHandler::_AddGlobalWritableFile(): "
1057 			"writable directory, recursion done\n");
1058 		return;
1059 	}
1060 
1061 	// get the package the target file originated from
1062 	BString originalPackage;
1063 	if (BNode(&targetDirectory, targetName).ReadAttrString(
1064 			kPackageFileAttribute, &originalPackage) != B_OK) {
1065 		// Can't determine the original package. The user must handle this
1066 		// manually.
1067 		PRINT("Volume::CommitTransactionHandler::_AddGlobalWritableFile(): "
1068 			"failed to get SYS:PACKAGE attribute for \"%s\", can't tell if "
1069 			"file needs to be updated\n",
1070 			targetName);
1071 		if (updateType != B_WRITABLE_FILE_UPDATE_TYPE_KEEP_OLD) {
1072 			_AddIssue(TransactionIssueBuilder(
1073 					BTransactionIssue::B_WRITABLE_FILE_NO_PACKAGE_ATTRIBUTE)
1074 				.SetPath1(FSUtils::Entry(targetDirectory, targetName)));
1075 		}
1076 		return;
1077 	}
1078 
1079 	// If that's our package, we're happy.
1080 	if (originalPackage == package->RevisionedNameThrows()) {
1081 		PRINT("Volume::CommitTransactionHandler::_AddGlobalWritableFile(): "
1082 			"file \"%s\" tagged with same package version we're activating\n",
1083 			targetName);
1084 		return;
1085 	}
1086 
1087 	// Check, whether the writable-files directory for the original package
1088 	// exists.
1089 	BString originalRelativeSourcePath = BString().SetToFormat("%s/%s",
1090 		originalPackage.String(), relativeSourcePath.ToCString());
1091 	if (originalRelativeSourcePath.IsEmpty())
1092 		throw std::bad_alloc();
1093 
1094 	struct stat originalPackageStat;
1095 	error = fWritableFilesDirectory.GetStatFor(originalRelativeSourcePath,
1096 		&originalPackageStat);
1097 	if (error != B_OK
1098 		|| (sourceStat.st_mode & S_IFMT)
1099 			!= (originalPackageStat.st_mode & S_IFMT)) {
1100 		// Original entry doesn't exist (either we don't have the data from
1101 		// the original package or the entry really didn't exist) or its
1102 		// type differs from the expected one. The user must handle this
1103 		// manually.
1104 		PRINT("Volume::CommitTransactionHandler::_AddGlobalWritableFile(): "
1105 			"original \"%s\" doesn't exist or has other type\n",
1106 			_GetPath(FSUtils::Entry(fWritableFilesDirectory,
1107 					originalRelativeSourcePath),
1108 				originalRelativeSourcePath).String());
1109 		if (error != B_OK) {
1110 			_AddIssue(TransactionIssueBuilder(
1111 					BTransactionIssue
1112 						::B_WRITABLE_FILE_OLD_ORIGINAL_FILE_MISSING)
1113 				.SetPath1(FSUtils::Entry(targetDirectory, targetName))
1114 				.SetPath2(FSUtils::Entry(fWritableFilesDirectory,
1115 					originalRelativeSourcePath)));
1116 		} else {
1117 			_AddIssue(TransactionIssueBuilder(
1118 					BTransactionIssue
1119 						::B_WRITABLE_FILE_OLD_ORIGINAL_FILE_TYPE_MISMATCH)
1120 				.SetPath1(FSUtils::Entry(targetDirectory, targetName))
1121 				.SetPath2(FSUtils::Entry(fWritableFilesDirectory,
1122 					originalRelativeSourcePath)));
1123 		}
1124 		return;
1125 	}
1126 
1127 	if (S_ISREG(sourceStat.st_mode)) {
1128 		// compare file content
1129 		bool equal;
1130 		error = FSUtils::CompareFileContent(
1131 			FSUtils::Entry(fWritableFilesDirectory,
1132 				originalRelativeSourcePath),
1133 			FSUtils::Entry(targetDirectory, targetName),
1134 			equal);
1135 		// TODO: Merge support!
1136 		if (error != B_OK || !equal) {
1137 			// The comparison failed or the files differ. The user must
1138 			// handle this manually.
1139 			PRINT("Volume::CommitTransactionHandler::"
1140 				"_AddGlobalWritableFile(): "
1141 				"file comparison \"%s\" failed (%s) or files aren't equal\n",
1142 				targetName, strerror(error));
1143 			if (updateType != B_WRITABLE_FILE_UPDATE_TYPE_KEEP_OLD) {
1144 				if (error != B_OK) {
1145 					_AddIssue(TransactionIssueBuilder(
1146 							BTransactionIssue
1147 								::B_WRITABLE_FILE_COMPARISON_FAILED)
1148 						.SetPath1(FSUtils::Entry(targetDirectory, targetName))
1149 						.SetPath2(FSUtils::Entry(fWritableFilesDirectory,
1150 							originalRelativeSourcePath))
1151 						.SetSystemError(error));
1152 				} else {
1153 					_AddIssue(TransactionIssueBuilder(
1154 							BTransactionIssue
1155 								::B_WRITABLE_FILE_NOT_EQUAL)
1156 						.SetPath1(FSUtils::Entry(targetDirectory, targetName))
1157 						.SetPath2(FSUtils::Entry(fWritableFilesDirectory,
1158 							originalRelativeSourcePath)));
1159 				}
1160 			}
1161 			return;
1162 		}
1163 	} else {
1164 		// compare symlinks
1165 		bool equal;
1166 		error = FSUtils::CompareSymLinks(
1167 			FSUtils::Entry(fWritableFilesDirectory,
1168 				originalRelativeSourcePath),
1169 			FSUtils::Entry(targetDirectory, targetName),
1170 			equal);
1171 		if (error != B_OK || !equal) {
1172 			// The comparison failed or the symlinks differ. The user must
1173 			// handle this manually.
1174 			PRINT("Volume::CommitTransactionHandler::"
1175 				"_AddGlobalWritableFile(): "
1176 				"symlink comparison \"%s\" failed (%s) or symlinks aren't "
1177 				"equal\n", targetName, strerror(error));
1178 			if (updateType != B_WRITABLE_FILE_UPDATE_TYPE_KEEP_OLD) {
1179 				if (error != B_OK) {
1180 					_AddIssue(TransactionIssueBuilder(
1181 							BTransactionIssue
1182 								::B_WRITABLE_SYMLINK_COMPARISON_FAILED)
1183 						.SetPath1(FSUtils::Entry(targetDirectory, targetName))
1184 						.SetPath2(FSUtils::Entry(fWritableFilesDirectory,
1185 							originalRelativeSourcePath))
1186 						.SetSystemError(error));
1187 				} else {
1188 					_AddIssue(TransactionIssueBuilder(
1189 							BTransactionIssue
1190 								::B_WRITABLE_SYMLINK_NOT_EQUAL)
1191 						.SetPath1(FSUtils::Entry(targetDirectory, targetName))
1192 						.SetPath2(FSUtils::Entry(fWritableFilesDirectory,
1193 							originalRelativeSourcePath)));
1194 				}
1195 			}
1196 			return;
1197 		}
1198 	}
1199 
1200 	// Replace the existing file/symlink. We do that in two steps: First
1201 	// copy the new file to a neighoring location, then move-replace the
1202 	// old file.
1203 	BString tempTargetName;
1204 	tempTargetName.SetToFormat("%s.%s", targetName,
1205 		package->RevisionedNameThrows().String());
1206 	if (tempTargetName.IsEmpty())
1207 		throw std::bad_alloc();
1208 
1209 	// copy
1210 	FSTransaction::CreateOperation copyOperation(&fFSTransaction,
1211 		FSUtils::Entry(targetDirectory, tempTargetName));
1212 
1213 	error = BCopyEngine(BCopyEngine::UNLINK_DESTINATION).CopyEntry(
1214 		FSUtils::Entry(sourceDirectory, relativeSourcePath.Leaf()),
1215 		FSUtils::Entry(targetDirectory, tempTargetName));
1216 	if (error != B_OK) {
1217 		throw Exception(B_TRANSACTION_FAILED_TO_COPY_FILE)
1218 			.SetPath1(_GetPath(
1219 				FSUtils::Entry(sourceDirectory,
1220 					relativeSourcePath.Leaf()),
1221 				relativeSourcePath))
1222 			.SetPath2(_GetPath(
1223 				FSUtils::Entry(targetDirectory, tempTargetName),
1224 				tempTargetName))
1225 			.SetSystemError(error);
1226 	}
1227 
1228 	copyOperation.Finished();
1229 
1230 	// rename
1231 	FSTransaction::RemoveOperation renameOperation(&fFSTransaction,
1232 		FSUtils::Entry(targetDirectory, targetName),
1233 		FSUtils::Entry(fWritableFilesDirectory,
1234 			originalRelativeSourcePath));
1235 
1236 	BEntry targetEntry;
1237 	error = targetEntry.SetTo(&targetDirectory, tempTargetName);
1238 	if (error == B_OK)
1239 		error = targetEntry.Rename(targetName, true);
1240 	if (error != B_OK) {
1241 		throw Exception(B_TRANSACTION_FAILED_TO_MOVE_FILE)
1242 			.SetPath1(_GetPath(
1243 				FSUtils::Entry(targetDirectory, tempTargetName),
1244 				tempTargetName))
1245 			.SetPath2(targetName)
1246 			.SetSystemError(error);
1247 	}
1248 
1249 	renameOperation.Finished();
1250 	copyOperation.Unregister();
1251 }
1252 
1253 
1254 void
1255 CommitTransactionHandler::_RevertAddPackagesToActivate()
1256 {
1257 	if (fAddedPackages.empty() || fFirstBootProcessing)
1258 		return;
1259 
1260 	// open transaction directory
1261 	BDirectory transactionDirectory;
1262 	status_t error = transactionDirectory.SetTo(&fTransactionDirectoryRef);
1263 	if (error != B_OK) {
1264 		ERROR("failed to open transaction directory: %s\n",
1265 			strerror(error));
1266 	}
1267 
1268 	for (PackageSet::iterator it = fAddedPackages.begin();
1269 		it != fAddedPackages.end(); ++it) {
1270 		// remove package from the volume
1271 		Package* package = *it;
1272 
1273 		if (fPackagesAlreadyAdded.find(package)
1274 				!= fPackagesAlreadyAdded.end()) {
1275 			continue;
1276 		}
1277 
1278 		fVolumeState->RemovePackage(package);
1279 
1280 		if (transactionDirectory.InitCheck() != B_OK)
1281 			continue;
1282 
1283 		// get BEntry for the package
1284 		NotOwningEntryRef entryRef(package->EntryRef());
1285 		BEntry entry;
1286 		error = entry.SetTo(&entryRef);
1287 		if (error != B_OK) {
1288 			ERROR("failed to get entry for package \"%s\": %s\n",
1289 				package->FileName().String(), strerror(error));
1290 			continue;
1291 		}
1292 
1293 		// move entry
1294 		error = entry.MoveTo(&transactionDirectory);
1295 		if (error != B_OK) {
1296 			ERROR("failed to move new package \"%s\" back to transaction "
1297 				"directory: %s\n", package->FileName().String(),
1298 				strerror(error));
1299 			continue;
1300 		}
1301 
1302 		fPackageFileManager->PackageFileMoved(package->File(),
1303 			fTransactionDirectoryRef);
1304 		package->File()->IncrementEntryRemovedIgnoreLevel();
1305 	}
1306 }
1307 
1308 
1309 void
1310 CommitTransactionHandler::_RevertRemovePackagesToDeactivate()
1311 {
1312 	if (fRemovedPackages.empty() || fFirstBootProcessing)
1313 		return;
1314 
1315 	// open packages directory
1316 	BDirectory packagesDirectory;
1317 	status_t error
1318 		= packagesDirectory.SetTo(&fVolume->PackagesDirectoryRef());
1319 	if (error != B_OK) {
1320 		throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
1321 			.SetPath1("<packages>")
1322 			.SetSystemError(error);
1323 	}
1324 
1325 	for (PackageSet::iterator it = fRemovedPackages.begin();
1326 		it != fRemovedPackages.end(); ++it) {
1327 		Package* package = *it;
1328 		if (fPackagesAlreadyRemoved.find(package)
1329 				!= fPackagesAlreadyRemoved.end()) {
1330 			continue;
1331 		}
1332 
1333 		// get a BEntry for the package
1334 		BEntry entry;
1335 		status_t error = entry.SetTo(&fOldStateDirectory,
1336 			package->FileName());
1337 		if (error != B_OK) {
1338 			ERROR("failed to get entry for package \"%s\": %s\n",
1339 				package->FileName().String(), strerror(error));
1340 			continue;
1341 		}
1342 
1343 		// move entry
1344 		error = entry.MoveTo(&packagesDirectory);
1345 		if (error != B_OK) {
1346 			ERROR("failed to move old package \"%s\" back to packages "
1347 				"directory: %s\n", package->FileName().String(),
1348 				strerror(error));
1349 			continue;
1350 		}
1351 
1352 		fPackageFileManager->PackageFileMoved(package->File(),
1353 			fVolume->PackagesDirectoryRef());
1354 		package->File()->IncrementEntryCreatedIgnoreLevel();
1355 	}
1356 }
1357 
1358 
1359 void
1360 CommitTransactionHandler::_RevertUserGroupChanges()
1361 {
1362 	// delete users
1363 	for (StringSet::const_iterator it = fAddedUsers.begin();
1364 		it != fAddedUsers.end(); ++it) {
1365 		std::string commandLine("userdel ");
1366 		commandLine += FSUtils::ShellEscapeString(it->c_str()).String();
1367 		if (system(commandLine.c_str()) != 0)
1368 			ERROR("failed to remove user \"%s\"\n", it->c_str());
1369 	}
1370 
1371 	// delete groups
1372 	for (StringSet::const_iterator it = fAddedGroups.begin();
1373 		it != fAddedGroups.end(); ++it) {
1374 		std::string commandLine("groupdel ");
1375 		commandLine += FSUtils::ShellEscapeString(it->c_str()).String();
1376 		if (system(commandLine.c_str()) != 0)
1377 			ERROR("failed to remove group \"%s\"\n", it->c_str());
1378 	}
1379 }
1380 
1381 
1382 void
1383 CommitTransactionHandler::_RunPostInstallScripts()
1384 {
1385 	for (PackageSet::iterator it = fAddedPackages.begin();
1386 		it != fAddedPackages.end(); ++it) {
1387 		Package* package = *it;
1388 		fCurrentPackage = package;
1389 		const BStringList& scripts = package->Info().PostInstallScripts();
1390 		int32 count = scripts.CountStrings();
1391 		for (int32 i = 0; i < count; i++)
1392 			_RunPostOrPreScript(package, scripts.StringAt(i), true);
1393 	}
1394 
1395 	fCurrentPackage = NULL;
1396 }
1397 
1398 
1399 void
1400 CommitTransactionHandler::_RunPreUninstallScripts()
1401 {
1402 	// Note this runs in the correct order, so dependents get uninstalled before
1403 	// the packages they depend on.  No need for a reversed loop.
1404 	for (PackageSet::iterator it = fPackagesToDeactivate.begin();
1405 		it != fPackagesToDeactivate.end(); ++it) {
1406 		Package* package = *it;
1407 		fCurrentPackage = package;
1408 		const BStringList& scripts = package->Info().PreUninstallScripts();
1409 		int32 count = scripts.CountStrings();
1410 		for (int32 i = 0; i < count; i++)
1411 			_RunPostOrPreScript(package, scripts.StringAt(i), false);
1412 	}
1413 
1414 	fCurrentPackage = NULL;
1415 }
1416 
1417 
1418 void
1419 CommitTransactionHandler::_RunPostOrPreScript(Package* package,
1420 	const BString& script, bool postNotPre)
1421 {
1422 	const char *postOrPreInstallWording = postNotPre
1423 		? "post-installation" : "pre-uninstall";
1424 	BDirectory rootDir(&fVolume->RootDirectoryRef());
1425 	BPath scriptPath(&rootDir, script);
1426 	status_t error = scriptPath.InitCheck();
1427 	if (error != B_OK) {
1428 		ERROR("Volume::CommitTransactionHandler::_RunPostOrPreScript(): "
1429 			"failed get path of %s script \"%s\" of package "
1430 			"%s: %s\n",
1431 			postOrPreInstallWording, script.String(),
1432 			package->FileName().String(), strerror(error));
1433 		_AddIssue(TransactionIssueBuilder(postNotPre
1434 				? BTransactionIssue::B_POST_INSTALL_SCRIPT_NOT_FOUND
1435 				: BTransactionIssue::B_PRE_UNINSTALL_SCRIPT_NOT_FOUND)
1436 			.SetPath1(script)
1437 			.SetSystemError(error));
1438 		return;
1439 	}
1440 
1441 	errno = 0;
1442 	int result = system(scriptPath.Path());
1443 	if (result != 0) {
1444 		ERROR("Volume::CommitTransactionHandler::_RunPostOrPreScript(): "
1445 			"running %s script \"%s\" of package %s "
1446 			"failed: %d (errno: %s)\n",
1447 			postOrPreInstallWording, script.String(),
1448 			package->FileName().String(), result, strerror(errno));
1449 		if (result < 0 || result == 127) { // bash shell returns 127 on failure.
1450 			_AddIssue(TransactionIssueBuilder(postNotPre
1451 					? BTransactionIssue::B_STARTING_POST_INSTALL_SCRIPT_FAILED
1452 					: BTransactionIssue::B_STARTING_PRE_UNINSTALL_SCRIPT_FAILED)
1453 				.SetPath1(BString(scriptPath.Path()))
1454 				.SetSystemError(errno));
1455 		} else { // positive is an exit code from the script itself.
1456 			_AddIssue(TransactionIssueBuilder(postNotPre
1457 					? BTransactionIssue::B_POST_INSTALL_SCRIPT_FAILED
1458 					: BTransactionIssue::B_PRE_UNINSTALL_SCRIPT_FAILED)
1459 				.SetPath1(BString(scriptPath.Path()))
1460 				.SetExitCode(result));
1461 		}
1462 	}
1463 }
1464 
1465 
1466 void
1467 CommitTransactionHandler::_QueuePostInstallScripts()
1468 {
1469 	BDirectory adminDirectory;
1470 	status_t error = _OpenPackagesSubDirectory(
1471 		RelativePath(kAdminDirectoryName), true, adminDirectory);
1472 	if (error != B_OK) {
1473 		ERROR("Failed to open administrative directory: %s\n", strerror(error));
1474 		return;
1475 	}
1476 
1477 	BDirectory scriptsDirectory;
1478 	error = scriptsDirectory.SetTo(&adminDirectory, kQueuedScriptsDirectoryName);
1479 	if (error == B_ENTRY_NOT_FOUND)
1480 		error = adminDirectory.CreateDirectory(kQueuedScriptsDirectoryName, &scriptsDirectory);
1481 	if (error != B_OK) {
1482 		ERROR("Failed to open queued scripts directory: %s\n", strerror(error));
1483 		return;
1484 	}
1485 
1486 	BDirectory rootDir(&fVolume->RootDirectoryRef());
1487 	for (PackageSet::iterator it = fAddedPackages.begin();
1488 		it != fAddedPackages.end(); ++it) {
1489 		Package* package = *it;
1490 		const BStringList& scripts = package->Info().PostInstallScripts();
1491 		for (int32 i = 0; i < scripts.CountStrings(); ++i) {
1492 			BPath scriptPath(&rootDir, scripts.StringAt(i));
1493 			status_t error = scriptPath.InitCheck();
1494 			if (error != B_OK) {
1495 				ERROR("Can't find script: %s\n", scripts.StringAt(i).String());
1496 				continue;
1497 			}
1498 
1499 			// symlink to the script
1500 			BSymLink scriptLink;
1501 			scriptsDirectory.CreateSymLink(scriptPath.Leaf(),
1502 				scriptPath.Path(), &scriptLink);
1503 			if (scriptLink.InitCheck() != B_OK) {
1504 				ERROR("Creating symlink failed: %s\n", strerror(scriptLink.InitCheck()));
1505 				continue;
1506 			}
1507 		}
1508 	}
1509 }
1510 
1511 
1512 void
1513 CommitTransactionHandler::_ExtractPackageContent(Package* package,
1514 	const BStringList& contentPaths, BDirectory& targetDirectory,
1515 	BDirectory& _extractedFilesDirectory)
1516 {
1517 	// check whether the subdirectory already exists
1518 	BString targetName(package->RevisionedNameThrows());
1519 
1520 	BEntry targetEntry;
1521 	status_t error = targetEntry.SetTo(&targetDirectory, targetName);
1522 	if (error != B_OK) {
1523 		throw Exception(B_TRANSACTION_FAILED_TO_ACCESS_ENTRY)
1524 			.SetPath1(_GetPath(
1525 				FSUtils::Entry(targetDirectory, targetName),
1526 				targetName))
1527 			.SetPackageName(package->FileName())
1528 			.SetSystemError(error);
1529 	}
1530 	if (targetEntry.Exists()) {
1531 		// nothing to do -- the very same version of the package has already
1532 		// been extracted
1533 		error = _extractedFilesDirectory.SetTo(&targetDirectory,
1534 			targetName);
1535 		if (error != B_OK) {
1536 			throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
1537 				.SetPath1(_GetPath(
1538 					FSUtils::Entry(targetDirectory, targetName),
1539 					targetName))
1540 				.SetPackageName(package->FileName())
1541 				.SetSystemError(error);
1542 		}
1543 		return;
1544 	}
1545 
1546 	// create the subdirectory with a temporary name (remove, if it already
1547 	// exists)
1548 	BString temporaryTargetName = BString().SetToFormat("%s.tmp",
1549 		targetName.String());
1550 	if (temporaryTargetName.IsEmpty())
1551 		throw std::bad_alloc();
1552 
1553 	error = targetEntry.SetTo(&targetDirectory, temporaryTargetName);
1554 	if (error != B_OK) {
1555 		throw Exception(B_TRANSACTION_FAILED_TO_ACCESS_ENTRY)
1556 			.SetPath1(_GetPath(
1557 				FSUtils::Entry(targetDirectory, temporaryTargetName),
1558 				temporaryTargetName))
1559 			.SetPackageName(package->FileName())
1560 			.SetSystemError(error);
1561 	}
1562 
1563 	if (targetEntry.Exists()) {
1564 		// remove pre-existing
1565 		error = BRemoveEngine().RemoveEntry(FSUtils::Entry(targetEntry));
1566 		if (error != B_OK) {
1567 			throw Exception(B_TRANSACTION_FAILED_TO_REMOVE_DIRECTORY)
1568 				.SetPath1(_GetPath(
1569 					FSUtils::Entry(targetDirectory, temporaryTargetName),
1570 					temporaryTargetName))
1571 				.SetPackageName(package->FileName())
1572 				.SetSystemError(error);
1573 		}
1574 	}
1575 
1576 	BDirectory& subDirectory = _extractedFilesDirectory;
1577 	FSTransaction::CreateOperation createSubDirectoryOperation(
1578 		&fFSTransaction,
1579 		FSUtils::Entry(targetDirectory, temporaryTargetName));
1580 	error = targetDirectory.CreateDirectory(temporaryTargetName,
1581 		&subDirectory);
1582 	if (error != B_OK) {
1583 		throw Exception(B_TRANSACTION_FAILED_TO_CREATE_DIRECTORY)
1584 			.SetPath1(_GetPath(
1585 				FSUtils::Entry(targetDirectory, temporaryTargetName),
1586 				temporaryTargetName))
1587 			.SetPackageName(package->FileName())
1588 			.SetSystemError(error);
1589 	}
1590 
1591 	createSubDirectoryOperation.Finished();
1592 
1593 	// extract
1594 	NotOwningEntryRef packageRef(package->EntryRef());
1595 
1596 	int32 contentPathCount = contentPaths.CountStrings();
1597 	for (int32 i = 0; i < contentPathCount; i++) {
1598 		const char* contentPath = contentPaths.StringAt(i);
1599 
1600 		error = FSUtils::ExtractPackageContent(FSUtils::Entry(packageRef),
1601 			contentPath, FSUtils::Entry(subDirectory));
1602 		if (error != B_OK) {
1603 			throw Exception(B_TRANSACTION_FAILED_TO_EXTRACT_PACKAGE_FILE)
1604 				.SetPath1(contentPath)
1605 				.SetPackageName(package->FileName())
1606 				.SetSystemError(error);
1607 		}
1608 	}
1609 
1610 	// tag all entries with the package attribute
1611 	_TagPackageEntriesRecursively(subDirectory, targetName, true);
1612 
1613 	// rename the subdirectory
1614 	error = targetEntry.Rename(targetName);
1615 	if (error != B_OK) {
1616 		throw Exception(B_TRANSACTION_FAILED_TO_MOVE_FILE)
1617 			.SetPath1(_GetPath(
1618 				FSUtils::Entry(targetDirectory, temporaryTargetName),
1619 				temporaryTargetName))
1620 			.SetPath2(targetName)
1621 			.SetPackageName(package->FileName())
1622 			.SetSystemError(error);
1623 	}
1624 
1625 	// keep the directory, regardless of whether the transaction is rolled
1626 	// back
1627 	createSubDirectoryOperation.Unregister();
1628 }
1629 
1630 
1631 status_t
1632 CommitTransactionHandler::_OpenPackagesSubDirectory(const RelativePath& path,
1633 	bool create, BDirectory& _directory)
1634 {
1635 	// open the packages directory
1636 	BDirectory directory;
1637 	status_t error = directory.SetTo(&fVolume->PackagesDirectoryRef());
1638 	if (error != B_OK) {
1639 		ERROR("CommitTransactionHandler::_OpenPackagesSubDirectory(): failed "
1640 			"to open packages directory: %s\n", strerror(error));
1641 		RETURN_ERROR(error);
1642 	}
1643 
1644 	return FSUtils::OpenSubDirectory(directory, path, create, _directory);
1645 }
1646 
1647 
1648 status_t
1649 CommitTransactionHandler::_OpenPackagesFile(
1650 	const RelativePath& subDirectoryPath, const char* fileName, uint32 openMode,
1651 	BFile& _file, BEntry* _entry)
1652 {
1653 	BDirectory directory;
1654 	if (!subDirectoryPath.IsEmpty()) {
1655 		status_t error = _OpenPackagesSubDirectory(subDirectoryPath,
1656 			(openMode & B_CREATE_FILE) != 0, directory);
1657 		if (error != B_OK) {
1658 			ERROR("CommitTransactionHandler::_OpenPackagesFile(): failed to "
1659 				"open packages subdirectory \"%s\": %s\n",
1660 				subDirectoryPath.ToString().String(), strerror(error));
1661 			RETURN_ERROR(error);
1662 		}
1663 	} else {
1664 		status_t error = directory.SetTo(&fVolume->PackagesDirectoryRef());
1665 		if (error != B_OK) {
1666 			ERROR("CommitTransactionHandler::_OpenPackagesFile(): failed to "
1667 				"open packages directory: %s\n", strerror(error));
1668 			RETURN_ERROR(error);
1669 		}
1670 	}
1671 
1672 	BEntry stackEntry;
1673 	BEntry& entry = _entry != NULL ? *_entry : stackEntry;
1674 	status_t error = entry.SetTo(&directory, fileName);
1675 	if (error != B_OK) {
1676 		ERROR("CommitTransactionHandler::_OpenPackagesFile(): failed to get "
1677 			"entry for file: %s", strerror(error));
1678 		RETURN_ERROR(error);
1679 	}
1680 
1681 	return _file.SetTo(&entry, openMode);
1682 }
1683 
1684 
1685 void
1686 CommitTransactionHandler::_WriteActivationFile(
1687 	const RelativePath& directoryPath, const char* fileName,
1688 	const PackageSet& toActivate, const PackageSet& toDeactivate,
1689 	BEntry& _entry)
1690 {
1691 	// create the content
1692 	BString activationFileContent;
1693 	_CreateActivationFileContent(toActivate, toDeactivate,
1694 		activationFileContent);
1695 
1696 	// write the file
1697 	status_t error = _WriteTextFile(directoryPath, fileName,
1698 		activationFileContent, _entry);
1699 	if (error != B_OK) {
1700 		BString filePath = directoryPath.ToString() << '/' << fileName;
1701 		throw Exception(B_TRANSACTION_FAILED_TO_WRITE_ACTIVATION_FILE)
1702 			.SetPath1(_GetPath(
1703 				FSUtils::Entry(fVolume->PackagesDirectoryRef(), filePath),
1704 				filePath))
1705 			.SetSystemError(error);
1706 	}
1707 }
1708 
1709 
1710 void
1711 CommitTransactionHandler::_CreateActivationFileContent(
1712 	const PackageSet& toActivate, const PackageSet& toDeactivate,
1713 	BString& _content)
1714 {
1715 	BString activationFileContent;
1716 	for (PackageFileNameHashTable::Iterator it
1717 			= fVolumeState->ByFileNameIterator();
1718 		Package* package = it.Next();) {
1719 		if (package->IsActive()
1720 			&& toDeactivate.find(package) == toDeactivate.end()) {
1721 			int32 length = activationFileContent.Length();
1722 			activationFileContent << package->FileName() << '\n';
1723 			if (activationFileContent.Length()
1724 					< length + package->FileName().Length() + 1) {
1725 				throw Exception(B_TRANSACTION_NO_MEMORY);
1726 			}
1727 		}
1728 	}
1729 
1730 	for (PackageSet::const_iterator it = toActivate.begin();
1731 		it != toActivate.end(); ++it) {
1732 		Package* package = *it;
1733 		int32 length = activationFileContent.Length();
1734 		activationFileContent << package->FileName() << '\n';
1735 		if (activationFileContent.Length()
1736 				< length + package->FileName().Length() + 1) {
1737 			throw Exception(B_TRANSACTION_NO_MEMORY);
1738 		}
1739 	}
1740 
1741 	_content = activationFileContent;
1742 }
1743 
1744 
1745 status_t
1746 CommitTransactionHandler::_WriteTextFile(const RelativePath& directoryPath,
1747 	const char* fileName, const BString& content, BEntry& _entry)
1748 {
1749 	BFile file;
1750 	status_t error = _OpenPackagesFile(directoryPath,
1751 		fileName, B_READ_WRITE | B_CREATE_FILE | B_ERASE_FILE, file, &_entry);
1752 	if (error != B_OK) {
1753 		ERROR("CommitTransactionHandler::_WriteTextFile(): failed to create "
1754 			"file \"%s/%s\": %s\n", directoryPath.ToString().String(), fileName,
1755 			strerror(error));
1756 		return error;
1757 	}
1758 
1759 	ssize_t bytesWritten = file.Write(content.String(),
1760 		content.Length());
1761 	if (bytesWritten < 0) {
1762 		ERROR("CommitTransactionHandler::_WriteTextFile(): failed to write "
1763 			"file \"%s/%s\": %s\n", directoryPath.ToString().String(), fileName,
1764 			strerror(bytesWritten));
1765 		return bytesWritten;
1766 	}
1767 
1768 	return B_OK;
1769 }
1770 
1771 
1772 void
1773 CommitTransactionHandler::_ChangePackageActivation(
1774 	const PackageSet& packagesToActivate,
1775 	const PackageSet& packagesToDeactivate)
1776 {
1777 	INFORM("CommitTransactionHandler::_ChangePackageActivation(): activating "
1778 		"%zu, deactivating %zu packages\n", packagesToActivate.size(),
1779 		packagesToDeactivate.size());
1780 
1781 	// write the temporary package activation file
1782 	BEntry activationFileEntry;
1783 	_WriteActivationFile(RelativePath(kAdminDirectoryName),
1784 		kTemporaryActivationFileName, packagesToActivate, packagesToDeactivate,
1785 		activationFileEntry);
1786 
1787 	// notify packagefs
1788 	if (fVolumeStateIsActive) {
1789 		_ChangePackageActivationIOCtl(packagesToActivate, packagesToDeactivate);
1790 	} else {
1791 		// TODO: Notify packagefs that active packages have been moved or do
1792 		// node monitoring in packagefs!
1793 	}
1794 
1795 	// rename the temporary activation file to the final file
1796 	status_t error = activationFileEntry.Rename(kActivationFileName, true);
1797 	if (error != B_OK) {
1798 		throw Exception(B_TRANSACTION_FAILED_TO_MOVE_FILE)
1799 			.SetPath1(_GetPath(
1800 				FSUtils::Entry(activationFileEntry),
1801 				activationFileEntry.Name()))
1802 			.SetPath2(kActivationFileName)
1803 			.SetSystemError(error);
1804 
1805 // TODO: We should probably try to revert the activation changes, though that
1806 // will fail, if this method has been called in response to node monitoring
1807 // events. Alternatively moving the package activation file could be made part
1808 // of the ioctl(), since packagefs should be able to undo package changes until
1809 // the very end, unless running out of memory. In the end the situation would be
1810 // bad anyway, though, since the activation file may refer to removed packages
1811 // and things would be in an inconsistent state after rebooting.
1812 	}
1813 
1814 	// Update our state, i.e. remove deactivated packages and mark activated
1815 	// packages accordingly.
1816 	fVolumeState->ActivationChanged(packagesToActivate, packagesToDeactivate);
1817 }
1818 
1819 
1820 void
1821 CommitTransactionHandler::_ChangePackageActivationIOCtl(
1822 	const PackageSet& packagesToActivate,
1823 	const PackageSet& packagesToDeactivate)
1824 {
1825 	// compute the size of the allocation we need for the activation change
1826 	// request
1827 	int32 itemCount = packagesToActivate.size() + packagesToDeactivate.size();
1828 	size_t requestSize = sizeof(PackageFSActivationChangeRequest)
1829 		+ itemCount * sizeof(PackageFSActivationChangeItem);
1830 
1831 	for (PackageSet::iterator it = packagesToActivate.begin();
1832 		 it != packagesToActivate.end(); ++it) {
1833 		requestSize += (*it)->FileName().Length() + 1;
1834 	}
1835 
1836 	for (PackageSet::iterator it = packagesToDeactivate.begin();
1837 		 it != packagesToDeactivate.end(); ++it) {
1838 		requestSize += (*it)->FileName().Length() + 1;
1839 	}
1840 
1841 	// allocate and prepare the request
1842 	PackageFSActivationChangeRequest* request
1843 		= (PackageFSActivationChangeRequest*)malloc(requestSize);
1844 	if (request == NULL)
1845 		throw Exception(B_TRANSACTION_NO_MEMORY);
1846 	MemoryDeleter requestDeleter(request);
1847 
1848 	request->itemCount = itemCount;
1849 
1850 	PackageFSActivationChangeItem* item = &request->items[0];
1851 	char* nameBuffer = (char*)(item + itemCount);
1852 
1853 	for (PackageSet::iterator it = packagesToActivate.begin();
1854 		it != packagesToActivate.end(); ++it, item++) {
1855 		_FillInActivationChangeItem(item, PACKAGE_FS_ACTIVATE_PACKAGE, *it,
1856 			nameBuffer);
1857 	}
1858 
1859 	for (PackageSet::iterator it = packagesToDeactivate.begin();
1860 		it != packagesToDeactivate.end(); ++it, item++) {
1861 		_FillInActivationChangeItem(item, PACKAGE_FS_DEACTIVATE_PACKAGE, *it,
1862 			nameBuffer);
1863 	}
1864 
1865 	// issue the request
1866 	FileDescriptorCloser fd(fVolume->OpenRootDirectory());
1867 	if (!fd.IsSet()) {
1868 		throw Exception(B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY)
1869 			.SetPath1(_GetPath(
1870 				FSUtils::Entry(fVolume->RootDirectoryRef()),
1871 				"<packagefs root>"))
1872 			.SetSystemError(fd.Get());
1873 	}
1874 
1875 	if (ioctl(fd.Get(), PACKAGE_FS_OPERATION_CHANGE_ACTIVATION, request,
1876 		requestSize) != 0) {
1877 // TODO: We need more error information and error handling!
1878 		throw Exception(B_TRANSACTION_FAILED_TO_CHANGE_PACKAGE_ACTIVATION)
1879 			.SetSystemError(errno);
1880 	}
1881 }
1882 
1883 
1884 void
1885 CommitTransactionHandler::_FillInActivationChangeItem(
1886 	PackageFSActivationChangeItem* item, PackageFSActivationChangeType type,
1887 	Package* package, char*& nameBuffer)
1888 {
1889 	item->type = type;
1890 	item->packageDeviceID = package->NodeRef().device;
1891 	item->packageNodeID = package->NodeRef().node;
1892 	item->nameLength = package->FileName().Length();
1893 	item->parentDeviceID = fVolume->PackagesDeviceID();
1894 	item->parentDirectoryID = fVolume->PackagesDirectoryID();
1895 	item->name = nameBuffer;
1896 	strcpy(nameBuffer, package->FileName());
1897 	nameBuffer += package->FileName().Length() + 1;
1898 }
1899 
1900 
1901 bool
1902 CommitTransactionHandler::_IsSystemPackage(Package* package)
1903 {
1904 	// package name should be "haiku[_<arch>]"
1905 	const BString& name = package->Info().Name();
1906 	if (!name.StartsWith("haiku"))
1907 		return false;
1908 	if (name.Length() == 5)
1909 		return true;
1910 	if (name[5] != '_')
1911 		return false;
1912 
1913 	BPackageArchitecture architecture;
1914 	return BPackageInfo::GetArchitectureByName(name.String() + 6, architecture)
1915 		== B_OK;
1916 }
1917 
1918 
1919 void
1920 CommitTransactionHandler::_AddIssue(const TransactionIssueBuilder& builder)
1921 {
1922 	fResult.AddIssue(builder.BuildIssue(fCurrentPackage));
1923 }
1924 
1925 
1926 /*static*/ BString
1927 CommitTransactionHandler::_GetPath(const FSUtils::Entry& entry,
1928 	const BString& fallback)
1929 {
1930 	BString path = entry.Path();
1931 	return path.IsEmpty() ? fallback : path;
1932 }
1933 
1934 
1935 /*static*/ void
1936 CommitTransactionHandler::_TagPackageEntriesRecursively(BDirectory& directory,
1937 	const BString& value, bool nonDirectoriesOnly)
1938 {
1939 	char buffer[offsetof(struct dirent, d_name) + B_FILE_NAME_LENGTH];
1940 	dirent *entry = (dirent*)buffer;
1941 	while (directory.GetNextDirents(entry, sizeof(buffer), 1) == 1) {
1942 		if (strcmp(entry->d_name, ".") == 0
1943 			|| strcmp(entry->d_name, "..") == 0) {
1944 			continue;
1945 		}
1946 
1947 		// determine type
1948 		struct stat st;
1949 		status_t error = directory.GetStatFor(entry->d_name, &st);
1950 		if (error != B_OK) {
1951 			throw Exception(B_TRANSACTION_FAILED_TO_ACCESS_ENTRY)
1952 				.SetPath1(_GetPath(
1953 					FSUtils::Entry(directory, entry->d_name),
1954 					entry->d_name))
1955 				.SetSystemError(error);
1956 		}
1957 		bool isDirectory = S_ISDIR(st.st_mode);
1958 
1959 		// open the node and set the attribute
1960 		BNode stackNode;
1961 		BDirectory stackDirectory;
1962 		BNode* node;
1963 		if (isDirectory) {
1964 			node = &stackDirectory;
1965 			error = stackDirectory.SetTo(&directory, entry->d_name);
1966 		} else {
1967 			node = &stackNode;
1968 			error = stackNode.SetTo(&directory, entry->d_name);
1969 		}
1970 
1971 		if (error != B_OK) {
1972 			throw Exception(isDirectory
1973 					? B_TRANSACTION_FAILED_TO_OPEN_DIRECTORY
1974 					: B_TRANSACTION_FAILED_TO_OPEN_FILE)
1975 				.SetPath1(_GetPath(
1976 					FSUtils::Entry(directory, entry->d_name),
1977 					entry->d_name))
1978 				.SetSystemError(error);
1979 		}
1980 
1981 		if (!isDirectory || !nonDirectoriesOnly) {
1982 			error = node->WriteAttrString(kPackageFileAttribute, &value);
1983 			if (error != B_OK) {
1984 				throw Exception(B_TRANSACTION_FAILED_TO_WRITE_FILE_ATTRIBUTE)
1985 					.SetPath1(_GetPath(
1986 						FSUtils::Entry(directory, entry->d_name),
1987 						entry->d_name))
1988 					.SetSystemError(error);
1989 			}
1990 		}
1991 
1992 		// recurse
1993 		if (isDirectory) {
1994 			_TagPackageEntriesRecursively(stackDirectory, value,
1995 				nonDirectoriesOnly);
1996 		}
1997 	}
1998 }
1999 
2000 
2001 /*static*/ status_t
2002 CommitTransactionHandler::_AssertEntriesAreEqual(const BEntry& entry,
2003 	const BDirectory* directory)
2004 {
2005 	BFile a;
2006 	status_t status = a.SetTo(&entry, B_READ_ONLY);
2007 	if (status != B_OK)
2008 		return status;
2009 
2010 	BFile b;
2011 	status = b.SetTo(directory, entry.Name(), B_READ_ONLY);
2012 	if (status != B_OK)
2013 		return status;
2014 
2015 	off_t aSize;
2016 	status = a.GetSize(&aSize);
2017 	if (status != B_OK)
2018 		return status;
2019 
2020 	off_t bSize;
2021 	status = b.GetSize(&bSize);
2022 	if (status != B_OK)
2023 		return status;
2024 
2025 	if (aSize != bSize)
2026 		return B_FILE_EXISTS;
2027 
2028 	const size_t bufferSize = 4096;
2029 	uint8 aBuffer[bufferSize];
2030 	uint8 bBuffer[bufferSize];
2031 
2032 	while (aSize > 0) {
2033 		ssize_t aRead = a.Read(aBuffer, bufferSize);
2034 		ssize_t bRead = b.Read(bBuffer, bufferSize);
2035 		if (aRead < 0 || aRead != bRead)
2036 			return B_FILE_EXISTS;
2037 		if (memcmp(aBuffer, bBuffer, aRead) != 0)
2038 			return B_FILE_EXISTS;
2039 		aSize -= aRead;
2040 	}
2041 
2042 	INFORM("CommitTransactionHandler::_AssertEntriesAreEqual(): "
2043 		"Package file '%s' already exists in target folder "
2044 		"with equal contents\n", entry.Name());
2045 	return B_OK;
2046 }
2047