]> git.saurik.com Git - apt-legacy.git/blame - apt-pkg/cdrom.cc
Break the ABI and fix the package limit.
[apt-legacy.git] / apt-pkg / cdrom.cc
CommitLineData
da6ee469
JF
1/*
2 */
3
da6ee469
JF
4#include<apt-pkg/init.h>
5#include<apt-pkg/error.h>
6#include<apt-pkg/cdromutl.h>
7#include<apt-pkg/strutl.h>
8#include<apt-pkg/cdrom.h>
9#include<sstream>
10#include<fstream>
11#include<config.h>
12#include<apti18n.h>
13#include <sys/stat.h>
14#include <fcntl.h>
15#include <dirent.h>
16#include <unistd.h>
17#include <stdio.h>
00ec24d0 18#include <algorithm>
0e5943eb 19#include <dlfcn.h>
da6ee469
JF
20
21#include "indexcopy.h"
22
23using namespace std;
24
25// FindPackages - Find the package files on the CDROM /*{{{*/
26// ---------------------------------------------------------------------
27/* We look over the cdrom for package files. This is a recursive
28 search that short circuits when it his a package file in the dir.
29 This speeds it up greatly as the majority of the size is in the
30 binary-* sub dirs. */
00ec24d0
JF
31bool pkgCdrom::FindPackages(string CD,
32 vector<string> &List,
33 vector<string> &SList,
34 vector<string> &SigList,
35 vector<string> &TransList,
da6ee469
JF
36 string &InfoDir, pkgCdromStatus *log,
37 unsigned int Depth)
38{
39 static ino_t Inodes[9];
00ec24d0 40 DIR *D;
da6ee469
JF
41
42 // if we have a look we "pulse" now
43 if(log)
44 log->Update();
45
46 if (Depth >= 7)
47 return true;
48
49 if (CD[CD.length()-1] != '/')
50 CD += '/';
51
52 if (chdir(CD.c_str()) != 0)
53 return _error->Errno("chdir","Unable to change to %s",CD.c_str());
54
55 // Look for a .disk subdirectory
56 struct stat Buf;
57 if (stat(".disk",&Buf) == 0)
58 {
59 if (InfoDir.empty() == true)
60 InfoDir = CD + ".disk/";
61 }
62
63 // Don't look into directories that have been marked to ingore.
64 if (stat(".aptignr",&Buf) == 0)
65 return true;
66
67
68 /* Check _first_ for a signature file as apt-cdrom assumes that all files
69 under a Packages/Source file are in control of that file and stops
70 the scanning
71 */
72 if (stat("Release.gpg",&Buf) == 0)
73 {
74 SigList.push_back(CD);
75 }
76 /* Aha! We found some package files. We assume that everything under
77 this dir is controlled by those package files so we don't look down
78 anymore */
79 if (stat("Packages",&Buf) == 0 || stat("Packages.gz",&Buf) == 0)
80 {
81 List.push_back(CD);
82
83 // Continue down if thorough is given
84 if (_config->FindB("APT::CDROM::Thorough",false) == false)
85 return true;
86 }
87 if (stat("Sources.gz",&Buf) == 0 || stat("Sources",&Buf) == 0)
88 {
89 SList.push_back(CD);
90
91 // Continue down if thorough is given
92 if (_config->FindB("APT::CDROM::Thorough",false) == false)
93 return true;
94 }
00ec24d0
JF
95
96 // see if we find translatin indexes
97 if (stat("i18n",&Buf) == 0)
98 {
99 D = opendir("i18n");
100 for (struct dirent *Dir = readdir(D); Dir != 0; Dir = readdir(D))
101 {
102 if(strstr(Dir->d_name,"Translation") != NULL)
103 {
104 if (_config->FindB("Debug::aptcdrom",false) == true)
105 std::clog << "found translations: " << Dir->d_name << "\n";
106 string file = Dir->d_name;
107 if(file.substr(file.size()-3,file.size()) == ".gz")
108 file = file.substr(0,file.size()-3);
109 TransList.push_back(CD+"i18n/"+ file);
110 }
111 }
112 closedir(D);
113 }
114
da6ee469 115
00ec24d0 116 D = opendir(".");
da6ee469
JF
117 if (D == 0)
118 return _error->Errno("opendir","Unable to read %s",CD.c_str());
119
120 // Run over the directory
121 for (struct dirent *Dir = readdir(D); Dir != 0; Dir = readdir(D))
122 {
123 // Skip some files..
124 if (strcmp(Dir->d_name,".") == 0 ||
125 strcmp(Dir->d_name,"..") == 0 ||
126 //strcmp(Dir->d_name,"source") == 0 ||
127 strcmp(Dir->d_name,".disk") == 0 ||
128 strcmp(Dir->d_name,"experimental") == 0 ||
129 strcmp(Dir->d_name,"binary-all") == 0 ||
130 strcmp(Dir->d_name,"debian-installer") == 0)
131 continue;
132
133 // See if the name is a sub directory
134 struct stat Buf;
135 if (stat(Dir->d_name,&Buf) != 0)
136 continue;
137
138 if (S_ISDIR(Buf.st_mode) == 0)
139 continue;
140
141 unsigned int I;
142 for (I = 0; I != Depth; I++)
143 if (Inodes[I] == Buf.st_ino)
144 break;
145 if (I != Depth)
146 continue;
147
148 // Store the inodes weve seen
149 Inodes[Depth] = Buf.st_ino;
150
151 // Descend
00ec24d0 152 if (FindPackages(CD + Dir->d_name,List,SList,SigList,TransList,InfoDir,log,Depth+1) == false)
da6ee469
JF
153 break;
154
155 if (chdir(CD.c_str()) != 0)
156 return _error->Errno("chdir","Unable to change to %s",CD.c_str());
157 };
158
159 closedir(D);
160
161 return !_error->PendingError();
162}
0e5943eb 163 /*}}}*/
da6ee469
JF
164// Score - We compute a 'score' for a path /*{{{*/
165// ---------------------------------------------------------------------
166/* Paths are scored based on how close they come to what I consider
167 normal. That is ones that have 'dist' 'stable' 'testing' will score
168 higher than ones without. */
169int pkgCdrom::Score(string Path)
170{
171 int Res = 0;
172 if (Path.find("stable/") != string::npos)
173 Res += 29;
174 if (Path.find("/binary-") != string::npos)
175 Res += 20;
176 if (Path.find("testing/") != string::npos)
177 Res += 28;
178 if (Path.find("unstable/") != string::npos)
179 Res += 27;
180 if (Path.find("/dists/") != string::npos)
181 Res += 40;
182 if (Path.find("/main/") != string::npos)
183 Res += 20;
184 if (Path.find("/contrib/") != string::npos)
185 Res += 20;
186 if (Path.find("/non-free/") != string::npos)
187 Res += 20;
188 if (Path.find("/non-US/") != string::npos)
189 Res += 20;
190 if (Path.find("/source/") != string::npos)
191 Res += 10;
192 if (Path.find("/debian/") != string::npos)
193 Res -= 10;
194
195 // check for symlinks in the patch leading to the actual file
196 // a symlink gets a big penalty
197 struct stat Buf;
198 string statPath = flNotFile(Path);
199 string cdromPath = _config->FindDir("Acquire::cdrom::mount","/cdrom/");
200 while(statPath != cdromPath && statPath != "./") {
201 statPath.resize(statPath.size()-1); // remove the trailing '/'
202 if (lstat(statPath.c_str(),&Buf) == 0) {
203 if(S_ISLNK(Buf.st_mode)) {
204 Res -= 60;
205 break;
206 }
207 }
208 statPath = flNotFile(statPath); // descent
209 }
210
211 return Res;
212}
da6ee469
JF
213 /*}}}*/
214// DropBinaryArch - Dump dirs with a string like /binary-<foo>/ /*{{{*/
215// ---------------------------------------------------------------------
216/* Here we drop everything that is not this machines arch */
217bool pkgCdrom::DropBinaryArch(vector<string> &List)
218{
219 char S[300];
220 snprintf(S,sizeof(S),"/binary-%s/",
221 _config->Find("Apt::Architecture").c_str());
222
223 for (unsigned int I = 0; I < List.size(); I++)
224 {
225 const char *Str = List[I].c_str();
226
227 const char *Res;
228 if ((Res = strstr(Str,"/binary-")) == 0)
229 continue;
230
231 // Weird, remove it.
232 if (strlen(Res) < strlen(S))
233 {
234 List.erase(List.begin() + I);
235 I--;
236 continue;
237 }
238
239 // See if it is our arch
240 if (stringcmp(Res,Res + strlen(S),S) == 0)
241 continue;
242
243 // Erase it
244 List.erase(List.begin() + I);
245 I--;
246 }
247
248 return true;
249}
0e5943eb 250 /*}}}*/
da6ee469
JF
251// DropRepeats - Drop repeated files resulting from symlinks /*{{{*/
252// ---------------------------------------------------------------------
253/* Here we go and stat every file that we found and strip dup inodes. */
254bool pkgCdrom::DropRepeats(vector<string> &List,const char *Name)
255{
256 // Get a list of all the inodes
257 ino_t *Inodes = new ino_t[List.size()];
258 for (unsigned int I = 0; I != List.size(); I++)
259 {
260 struct stat Buf;
261 if (stat((List[I] + Name).c_str(),&Buf) != 0 &&
262 stat((List[I] + Name + ".gz").c_str(),&Buf) != 0)
263 _error->Errno("stat","Failed to stat %s%s",List[I].c_str(),
264 Name);
265 Inodes[I] = Buf.st_ino;
266 }
267
268 if (_error->PendingError() == true)
269 return false;
270
271 // Look for dups
272 for (unsigned int I = 0; I != List.size(); I++)
273 {
274 for (unsigned int J = I+1; J < List.size(); J++)
275 {
276 // No match
277 if (Inodes[J] != Inodes[I])
278 continue;
279
280 // We score the two paths.. and erase one
281 int ScoreA = Score(List[I]);
282 int ScoreB = Score(List[J]);
283 if (ScoreA < ScoreB)
284 {
285 List[I] = string();
286 break;
287 }
288
289 List[J] = string();
290 }
291 }
292
293 // Wipe erased entries
294 for (unsigned int I = 0; I < List.size();)
295 {
296 if (List[I].empty() == false)
297 I++;
298 else
299 List.erase(List.begin()+I);
300 }
301
302 return true;
303}
304 /*}}}*/
da6ee469
JF
305// ReduceSourceList - Takes the path list and reduces it /*{{{*/
306// ---------------------------------------------------------------------
307/* This takes the list of source list expressed entires and collects
308 similar ones to form a single entry for each dist */
309void pkgCdrom::ReduceSourcelist(string CD,vector<string> &List)
310{
311 sort(List.begin(),List.end());
312
313 // Collect similar entries
314 for (vector<string>::iterator I = List.begin(); I != List.end(); I++)
315 {
316 // Find a space..
317 string::size_type Space = (*I).find(' ');
318 if (Space == string::npos)
319 continue;
320 string::size_type SSpace = (*I).find(' ',Space + 1);
321 if (SSpace == string::npos)
322 continue;
323
324 string Word1 = string(*I,Space,SSpace-Space);
325 string Prefix = string(*I,0,Space);
326 for (vector<string>::iterator J = List.begin(); J != I; J++)
327 {
328 // Find a space..
329 string::size_type Space2 = (*J).find(' ');
330 if (Space2 == string::npos)
331 continue;
332 string::size_type SSpace2 = (*J).find(' ',Space2 + 1);
333 if (SSpace2 == string::npos)
334 continue;
335
336 if (string(*J,0,Space2) != Prefix)
337 continue;
338 if (string(*J,Space2,SSpace2-Space2) != Word1)
339 continue;
340
341 *J += string(*I,SSpace);
342 *I = string();
343 }
344 }
345
346 // Wipe erased entries
347 for (unsigned int I = 0; I < List.size();)
348 {
349 if (List[I].empty() == false)
350 I++;
351 else
352 List.erase(List.begin()+I);
353 }
354}
355 /*}}}*/
356// WriteDatabase - Write the CDROM Database file /*{{{*/
357// ---------------------------------------------------------------------
358/* We rewrite the configuration class associated with the cdrom database. */
359bool pkgCdrom::WriteDatabase(Configuration &Cnf)
360{
361 string DFile = _config->FindFile("Dir::State::cdroms");
362 string NewFile = DFile + ".new";
363
364 unlink(NewFile.c_str());
365 ofstream Out(NewFile.c_str());
366 if (!Out)
367 return _error->Errno("ofstream::ofstream",
368 "Failed to open %s.new",DFile.c_str());
369
370 /* Write out all of the configuration directives by walking the
371 configuration tree */
372 const Configuration::Item *Top = Cnf.Tree(0);
373 for (; Top != 0;)
374 {
375 // Print the config entry
376 if (Top->Value.empty() == false)
377 Out << Top->FullTag() + " \"" << Top->Value << "\";" << endl;
378
379 if (Top->Child != 0)
380 {
381 Top = Top->Child;
382 continue;
383 }
384
385 while (Top != 0 && Top->Next == 0)
386 Top = Top->Parent;
387 if (Top != 0)
388 Top = Top->Next;
389 }
390
391 Out.close();
392
393 rename(DFile.c_str(),string(DFile + '~').c_str());
394 if (rename(NewFile.c_str(),DFile.c_str()) != 0)
395 return _error->Errno("rename","Failed to rename %s.new to %s",
396 DFile.c_str(),DFile.c_str());
397
398 return true;
399}
400 /*}}}*/
401// WriteSourceList - Write an updated sourcelist /*{{{*/
402// ---------------------------------------------------------------------
403/* This reads the old source list and copies it into the new one. It
404 appends the new CDROM entires just after the first block of comments.
405 This places them first in the file. It also removes any old entries
406 that were the same. */
407bool pkgCdrom::WriteSourceList(string Name,vector<string> &List,bool Source)
408{
409 if (List.size() == 0)
410 return true;
411
412 string File = _config->FindFile("Dir::Etc::sourcelist");
413
414 // Open the stream for reading
415 ifstream F((FileExists(File)?File.c_str():"/dev/null"),
416 ios::in );
417 if (!F != 0)
418 return _error->Errno("ifstream::ifstream","Opening %s",File.c_str());
419
420 string NewFile = File + ".new";
421 unlink(NewFile.c_str());
422 ofstream Out(NewFile.c_str());
423 if (!Out)
424 return _error->Errno("ofstream::ofstream",
425 "Failed to open %s.new",File.c_str());
426
427 // Create a short uri without the path
428 string ShortURI = "cdrom:[" + Name + "]/";
429 string ShortURI2 = "cdrom:" + Name + "/"; // For Compatibility
430
431 string Type;
432 if (Source == true)
433 Type = "deb-src";
434 else
435 Type = "deb";
436
437 char Buffer[300];
438 int CurLine = 0;
439 bool First = true;
440 while (F.eof() == false)
441 {
442 F.getline(Buffer,sizeof(Buffer));
443 CurLine++;
444 if (F.fail() && !F.eof())
445 return _error->Error(_("Line %u too long in source list %s."),
446 CurLine,File.c_str());
447 _strtabexpand(Buffer,sizeof(Buffer));
448 _strstrip(Buffer);
449
450 // Comment or blank
451 if (Buffer[0] == '#' || Buffer[0] == 0)
452 {
453 Out << Buffer << endl;
454 continue;
455 }
456
457 if (First == true)
458 {
459 for (vector<string>::iterator I = List.begin(); I != List.end(); I++)
460 {
461 string::size_type Space = (*I).find(' ');
462 if (Space == string::npos)
463 return _error->Error("Internal error");
464 Out << Type << " cdrom:[" << Name << "]/" << string(*I,0,Space) <<
465 " " << string(*I,Space+1) << endl;
466 }
467 }
468 First = false;
469
470 // Grok it
471 string cType;
472 string URI;
473 const char *C = Buffer;
474 if (ParseQuoteWord(C,cType) == false ||
475 ParseQuoteWord(C,URI) == false)
476 {
477 Out << Buffer << endl;
478 continue;
479 }
480
481 // Emit lines like this one
482 if (cType != Type || (string(URI,0,ShortURI.length()) != ShortURI &&
483 string(URI,0,ShortURI.length()) != ShortURI2))
484 {
485 Out << Buffer << endl;
486 continue;
487 }
488 }
489
490 // Just in case the file was empty
491 if (First == true)
492 {
493 for (vector<string>::iterator I = List.begin(); I != List.end(); I++)
494 {
495 string::size_type Space = (*I).find(' ');
496 if (Space == string::npos)
497 return _error->Error("Internal error");
498
499 Out << "deb cdrom:[" << Name << "]/" << string(*I,0,Space) <<
500 " " << string(*I,Space+1) << endl;
501 }
502 }
503
504 Out.close();
505
506 rename(File.c_str(),string(File + '~').c_str());
507 if (rename(NewFile.c_str(),File.c_str()) != 0)
508 return _error->Errno("rename","Failed to rename %s.new to %s",
509 File.c_str(),File.c_str());
510
511 return true;
512}
0e5943eb
JF
513 /*}}}*/
514bool pkgCdrom::Ident(string &ident, pkgCdromStatus *log) /*{{{*/
da6ee469
JF
515{
516 stringstream msg;
517
518 // Startup
519 string CDROM = _config->FindDir("Acquire::cdrom::mount","/cdrom/");
520 if (CDROM[0] == '.')
521 CDROM= SafeGetCWD() + '/' + CDROM;
522
523 if(log) {
524 msg.str("");
525 ioprintf(msg, _("Using CD-ROM mount point %s\nMounting CD-ROM\n"),
526 CDROM.c_str());
527 log->Update(msg.str());
528 }
529 if (MountCdrom(CDROM) == false)
530 return _error->Error("Failed to mount the cdrom.");
531
532 // Hash the CD to get an ID
533 if(log)
534 log->Update(_("Identifying.. "));
535
536
537 if (IdentCdrom(CDROM,ident) == false)
538 {
539 ident = "";
540 return false;
541 }
542
543 msg.str("");
544 ioprintf(msg, "[%s]\n",ident.c_str());
545 log->Update(msg.str());
546
547
548 // Read the database
549 Configuration Database;
550 string DFile = _config->FindFile("Dir::State::cdroms");
551 if (FileExists(DFile) == true)
552 {
553 if (ReadConfigFile(Database,DFile) == false)
554 return _error->Error("Unable to read the cdrom database %s",
555 DFile.c_str());
556 }
557 if(log) {
558 msg.str("");
00ec24d0
JF
559 ioprintf(msg, _("Stored label: %s\n"),
560 Database.Find("CD::"+ident).c_str());
da6ee469
JF
561 log->Update(msg.str());
562 }
00ec24d0
JF
563
564 // Unmount and finish
565 if (_config->FindB("APT::CDROM::NoMount",false) == false) {
566 log->Update(_("Unmounting CD-ROM...\n"), STEP_LAST);
567 UnmountCdrom(CDROM);
568 }
569
da6ee469
JF
570 return true;
571}
0e5943eb
JF
572 /*}}}*/
573bool pkgCdrom::Add(pkgCdromStatus *log) /*{{{*/
da6ee469
JF
574{
575 stringstream msg;
576
577 // Startup
578 string CDROM = _config->FindDir("Acquire::cdrom::mount","/cdrom/");
579 if (CDROM[0] == '.')
580 CDROM= SafeGetCWD() + '/' + CDROM;
581
582 if(log) {
583 log->SetTotal(STEP_LAST);
584 msg.str("");
585 ioprintf(msg, _("Using CD-ROM mount point %s\n"), CDROM.c_str());
586 log->Update(msg.str(), STEP_PREPARE);
587 }
588
589 // Read the database
590 Configuration Database;
591 string DFile = _config->FindFile("Dir::State::cdroms");
592 if (FileExists(DFile) == true)
593 {
594 if (ReadConfigFile(Database,DFile) == false)
595 return _error->Error("Unable to read the cdrom database %s",
596 DFile.c_str());
597 }
598
599 // Unmount the CD and get the user to put in the one they want
600 if (_config->FindB("APT::CDROM::NoMount",false) == false)
601 {
602 if(log)
603 log->Update(_("Unmounting CD-ROM\n"), STEP_UNMOUNT);
604 UnmountCdrom(CDROM);
605
606 if(log) {
607 log->Update(_("Waiting for disc...\n"), STEP_WAIT);
608 if(!log->ChangeCdrom()) {
609 // user aborted
610 return false;
611 }
612 }
613
614 // Mount the new CDROM
615 log->Update(_("Mounting CD-ROM...\n"), STEP_MOUNT);
616 if (MountCdrom(CDROM) == false)
617 return _error->Error("Failed to mount the cdrom.");
618 }
619
620 // Hash the CD to get an ID
621 if(log)
622 log->Update(_("Identifying.. "), STEP_IDENT);
623 string ID;
624 if (IdentCdrom(CDROM,ID) == false)
625 {
626 log->Update("\n");
627 return false;
628 }
629 if(log)
630 log->Update("["+ID+"]\n");
631
632 if(log)
633 log->Update(_("Scanning disc for index files..\n"),STEP_SCAN);
634
635 // Get the CD structure
636 vector<string> List;
637 vector<string> SourceList;
638 vector<string> SigList;
00ec24d0 639 vector<string> TransList;
da6ee469
JF
640 string StartDir = SafeGetCWD();
641 string InfoDir;
00ec24d0 642 if (FindPackages(CDROM,List,SourceList, SigList,TransList,InfoDir,log) == false)
da6ee469
JF
643 {
644 log->Update("\n");
645 return false;
646 }
647
648 chdir(StartDir.c_str());
649
650 if (_config->FindB("Debug::aptcdrom",false) == true)
651 {
652 cout << "I found (binary):" << endl;
653 for (vector<string>::iterator I = List.begin(); I != List.end(); I++)
654 cout << *I << endl;
655 cout << "I found (source):" << endl;
656 for (vector<string>::iterator I = SourceList.begin(); I != SourceList.end(); I++)
657 cout << *I << endl;
658 cout << "I found (Signatures):" << endl;
659 for (vector<string>::iterator I = SigList.begin(); I != SigList.end(); I++)
660 cout << *I << endl;
661 }
662
663 //log->Update(_("Cleaning package lists..."), STEP_CLEAN);
664
665 // Fix up the list
666 DropBinaryArch(List);
667 DropRepeats(List,"Packages");
668 DropRepeats(SourceList,"Sources");
669 DropRepeats(SigList,"Release.gpg");
00ec24d0 670 DropRepeats(TransList,"");
da6ee469
JF
671 if(log) {
672 msg.str("");
00ec24d0
JF
673 ioprintf(msg, _("Found %zu package indexes, %zu source indexes, "
674 "%zu translation indexes and %zu signatures\n"),
675 List.size(), SourceList.size(), TransList.size(),
676 SigList.size());
da6ee469
JF
677 log->Update(msg.str(), STEP_SCAN);
678 }
679
680 if (List.size() == 0 && SourceList.size() == 0)
681 {
00ec24d0
JF
682 if (_config->FindB("APT::CDROM::NoMount",false) == false)
683 UnmountCdrom(CDROM);
0e5943eb 684 return _error->Error(_("Unable to locate any package files, perhaps this is not a Debian Disc or the wrong architecture?"));
da6ee469
JF
685 }
686
687 // Check if the CD is in the database
688 string Name;
689 if (Database.Exists("CD::" + ID) == false ||
690 _config->FindB("APT::CDROM::Rename",false) == true)
691 {
692 // Try to use the CDs label if at all possible
693 if (InfoDir.empty() == false &&
694 FileExists(InfoDir + "/info") == true)
695 {
696 ifstream F(string(InfoDir + "/info").c_str());
697 if (!F == 0)
698 getline(F,Name);
699
700 if (Name.empty() == false)
701 {
702 // Escape special characters
703 string::iterator J = Name.begin();
704 for (; J != Name.end(); J++)
705 if (*J == '"' || *J == ']' || *J == '[')
706 *J = '_';
707
708 if(log) {
709 msg.str("");
00ec24d0 710 ioprintf(msg, _("Found label '%s'\n"), Name.c_str());
da6ee469
JF
711 log->Update(msg.str());
712 }
713 Database.Set("CD::" + ID + "::Label",Name);
714 }
715 }
716
717 if (_config->FindB("APT::CDROM::Rename",false) == true ||
718 Name.empty() == true)
719 {
720 if(!log)
721 {
00ec24d0
JF
722 if (_config->FindB("APT::CDROM::NoMount",false) == false)
723 UnmountCdrom(CDROM);
da6ee469
JF
724 return _error->Error("No disc name found and no way to ask for it");
725 }
726
727 while(true) {
728 if(!log->AskCdromName(Name)) {
729 // user canceld
730 return false;
731 }
732 cout << "Name: '" << Name << "'" << endl;
733
734 if (Name.empty() == false &&
735 Name.find('"') == string::npos &&
736 Name.find('[') == string::npos &&
737 Name.find(']') == string::npos)
738 break;
739 log->Update(_("That is not a valid name, try again.\n"));
740 }
741 }
742 }
743 else
744 Name = Database.Find("CD::" + ID);
745
746 // Escape special characters
747 string::iterator J = Name.begin();
748 for (; J != Name.end(); J++)
749 if (*J == '"' || *J == ']' || *J == '[')
750 *J = '_';
751
752 Database.Set("CD::" + ID,Name);
753 if(log) {
754 msg.str("");
755 ioprintf(msg, _("This disc is called: \n'%s'\n"), Name.c_str());
756 log->Update(msg.str());
757 }
758
759 log->Update(_("Copying package lists..."), STEP_COPY);
760 // take care of the signatures and copy them if they are ok
761 // (we do this before PackageCopy as it modifies "List" and "SourceList")
762 SigVerify SignVerify;
763 SignVerify.CopyAndVerify(CDROM, Name, SigList, List, SourceList);
764
765 // Copy the package files to the state directory
766 PackageCopy Copy;
767 SourceCopy SrcCopy;
00ec24d0 768 TranslationsCopy TransCopy;
da6ee469 769 if (Copy.CopyPackages(CDROM,Name,List, log) == false ||
00ec24d0
JF
770 SrcCopy.CopyPackages(CDROM,Name,SourceList, log) == false ||
771 TransCopy.CopyTranslations(CDROM,Name,TransList, log) == false)
da6ee469
JF
772 return false;
773
774 // reduce the List so that it takes less space in sources.list
775 ReduceSourcelist(CDROM,List);
776 ReduceSourcelist(CDROM,SourceList);
777
778 // Write the database and sourcelist
779 if (_config->FindB("APT::cdrom::NoAct",false) == false)
780 {
781 if (WriteDatabase(Database) == false)
782 return false;
783
784 if(log) {
785 log->Update(_("Writing new source list\n"), STEP_WRITE);
786 }
787 if (WriteSourceList(Name,List,false) == false ||
788 WriteSourceList(Name,SourceList,true) == false)
789 return false;
790 }
791
792 // Print the sourcelist entries
793 if(log)
794 log->Update(_("Source list entries for this disc are:\n"));
795
796 for (vector<string>::iterator I = List.begin(); I != List.end(); I++)
797 {
798 string::size_type Space = (*I).find(' ');
799 if (Space == string::npos)
800 {
00ec24d0
JF
801 if (_config->FindB("APT::CDROM::NoMount",false) == false)
802 UnmountCdrom(CDROM);
da6ee469
JF
803 return _error->Error("Internal error");
804 }
805
806 if(log) {
807 msg.str("");
808 msg << "deb cdrom:[" << Name << "]/" << string(*I,0,Space) <<
809 " " << string(*I,Space+1) << endl;
810 log->Update(msg.str());
811 }
812 }
813
814 for (vector<string>::iterator I = SourceList.begin(); I != SourceList.end(); I++)
815 {
816 string::size_type Space = (*I).find(' ');
817 if (Space == string::npos)
818 {
00ec24d0
JF
819 if (_config->FindB("APT::CDROM::NoMount",false) == false)
820 UnmountCdrom(CDROM);
da6ee469
JF
821 return _error->Error("Internal error");
822 }
823
824 if(log) {
825 msg.str("");
826 msg << "deb-src cdrom:[" << Name << "]/" << string(*I,0,Space) <<
827 " " << string(*I,Space+1) << endl;
828 log->Update(msg.str());
829 }
830 }
831
da6ee469
JF
832 // Unmount and finish
833 if (_config->FindB("APT::CDROM::NoMount",false) == false) {
00ec24d0 834 log->Update(_("Unmounting CD-ROM...\n"), STEP_LAST);
da6ee469
JF
835 UnmountCdrom(CDROM);
836 }
837
838 return true;
839}
0e5943eb
JF
840 /*}}}*/
841pkgUdevCdromDevices::pkgUdevCdromDevices() /*{{{*/
842 : libudev_handle(NULL)
843{
844
845}
846 /*}}}*/
847
848bool
849pkgUdevCdromDevices::Dlopen() /*{{{*/
850{
851 // alread open
852 if(libudev_handle != NULL)
853 return true;
854
855 // see if we can get libudev
856 void *h = ::dlopen("libudev.so.0", RTLD_LAZY);
857 if(h == NULL)
858 return false;
859
860 // get the pointers to the udev structs
861 libudev_handle = h;
862 udev_new = (udev* (*)(void)) dlsym(h, "udev_new");
863 udev_enumerate_add_match_property = (int (*)(udev_enumerate*, const char*, const char*))dlsym(h, "udev_enumerate_add_match_property");
864 udev_enumerate_scan_devices = (int (*)(udev_enumerate*))dlsym(h, "udev_enumerate_scan_devices");
865 udev_enumerate_get_list_entry = (udev_list_entry* (*)(udev_enumerate*))dlsym(h, "udev_enumerate_get_list_entry");
866 udev_device_new_from_syspath = (udev_device* (*)(udev*, const char*))dlsym(h, "udev_device_new_from_syspath");
867 udev_enumerate_get_udev = (udev* (*)(udev_enumerate*))dlsym(h, "udev_enumerate_get_udev");
868 udev_list_entry_get_name = (const char* (*)(udev_list_entry*))dlsym(h, "udev_list_entry_get_name");
869 udev_device_get_devnode = (const char* (*)(udev_device*))dlsym(h, "udev_device_get_devnode");
870 udev_enumerate_new = (udev_enumerate* (*)(udev*))dlsym(h, "udev_enumerate_new");
871 udev_list_entry_get_next = (udev_list_entry* (*)(udev_list_entry*))dlsym(h, "udev_list_entry_get_next");
872 udev_device_get_property_value = (const char* (*)(udev_device *, const char *))dlsym(h, "udev_device_get_property_value");
873
874 return true;
875}
876 /*}}}*/
877vector<CdromDevice>
878pkgUdevCdromDevices::Scan() /*{{{*/
879{
880 vector<CdromDevice> cdrom_devices;
881 struct udev_enumerate *enumerate;
882 struct udev_list_entry *l, *devices;
883 struct udev *udev_ctx;
884
885 if(libudev_handle == NULL)
886 return cdrom_devices;
887
888 udev_ctx = udev_new();
889 enumerate = udev_enumerate_new (udev_ctx);
890 udev_enumerate_add_match_property(enumerate, "ID_CDROM", "1");
891
892 udev_enumerate_scan_devices (enumerate);
893 devices = udev_enumerate_get_list_entry (enumerate);
894 for (l = devices; l != NULL; l = udev_list_entry_get_next (l))
895 {
896 CdromDevice cdrom;
897 struct udev_device *udevice;
898 udevice = udev_device_new_from_syspath (udev_enumerate_get_udev (enumerate), udev_list_entry_get_name (l));
899 if (udevice == NULL)
900 continue;
901 const char* devnode = udev_device_get_devnode(udevice);
902 const char* mountpath = udev_device_get_property_value(udevice, "FSTAB_DIR");
903
904 // fill in the struct
905 cdrom.DeviceName = string(devnode);
906 if (mountpath) {
907 cdrom.MountPath = mountpath;
908 string s = string(mountpath);
909 cdrom.Mounted = IsMounted(s);
910 } else {
911 cdrom.Mounted = false;
912 cdrom.MountPath = "";
913 }
914 cdrom_devices.push_back(cdrom);
915 }
916 return cdrom_devices;
917}
918 /*}}}*/
919
920pkgUdevCdromDevices::~pkgUdevCdromDevices() /*{{{*/
921{
922 if (libudev_handle != NULL)
923 dlclose(libudev_handle);
924}
925 /*}}}*/