]>
Commit | Line | Data |
---|---|---|
f46e7681 AL |
1 | // -*- mode: cpp; mode: fold -*- |
2 | // Description /*{{{*/ | |
b3d44315 | 3 | // $Id: cdrom.cc,v 1.20.2.1 2004/01/16 18:58:50 mdz Exp $ |
f46e7681 AL |
4 | /* ###################################################################### |
5 | ||
6 | CDROM URI method for APT | |
7 | ||
8 | ##################################################################### */ | |
9 | /*}}}*/ | |
10 | // Include Files /*{{{*/ | |
11 | #include <apt-pkg/acquire-method.h> | |
a6418a4b | 12 | #include <apt-pkg/cdrom.h> |
f46e7681 AL |
13 | #include <apt-pkg/cdromutl.h> |
14 | #include <apt-pkg/error.h> | |
15 | #include <apt-pkg/configuration.h> | |
16 | #include <apt-pkg/fileutl.h> | |
13e8426f | 17 | #include <apt-pkg/hashes.h> |
f46e7681 AL |
18 | |
19 | #include <sys/stat.h> | |
20 | #include <unistd.h> | |
8e372e79 | 21 | #include <dlfcn.h> |
076cc664 AL |
22 | |
23 | #include <iostream> | |
d77559ac | 24 | #include <apti18n.h> |
f46e7681 AL |
25 | /*}}}*/ |
26 | ||
076cc664 AL |
27 | using namespace std; |
28 | ||
f46e7681 AL |
29 | class CDROMMethod : public pkgAcqMethod |
30 | { | |
f631d1ba | 31 | bool DatabaseLoaded; |
76fe5db7 MV |
32 | bool Debug; |
33 | ||
5b76e7f2 | 34 | ::Configuration Database; |
f46e7681 | 35 | string CurrentID; |
8e5fc8f5 | 36 | string CDROM; |
70dbf5f8 | 37 | bool MountedByApt; |
76fe5db7 | 38 | pkgUdevCdromDevices UdevCdroms; |
8e372e79 | 39 | |
a6418a4b | 40 | bool IsCorrectCD(URI want, string MountPath); |
76fe5db7 | 41 | bool AutoDetectAndMount(URI); |
f46e7681 AL |
42 | virtual bool Fetch(FetchItem *Itm); |
43 | string GetID(string Name); | |
8e5fc8f5 | 44 | virtual void Exit(); |
8e372e79 | 45 | |
f46e7681 AL |
46 | public: |
47 | ||
48 | CDROMMethod(); | |
49 | }; | |
50 | ||
51 | // CDROMMethod::CDROMethod - Constructor /*{{{*/ | |
52 | // --------------------------------------------------------------------- | |
53 | /* */ | |
459681d3 AL |
54 | CDROMMethod::CDROMMethod() : pkgAcqMethod("1.0",SingleInstance | LocalOnly | |
55 | SendConfig | NeedsCleanup | | |
56 | Removable), | |
8e5fc8f5 | 57 | DatabaseLoaded(false), |
70dbf5f8 | 58 | MountedByApt(false) |
f46e7681 | 59 | { |
76fe5db7 | 60 | UdevCdroms.Dlopen(); |
f46e7681 AL |
61 | }; |
62 | /*}}}*/ | |
8e5fc8f5 AL |
63 | // CDROMMethod::Exit - Unmount the disc if necessary /*{{{*/ |
64 | // --------------------------------------------------------------------- | |
65 | /* */ | |
66 | void CDROMMethod::Exit() | |
8e372e79 MV |
67 | { |
68 | if (MountedByApt == true) | |
8e5fc8f5 AL |
69 | UnmountCdrom(CDROM); |
70 | } | |
71 | /*}}}*/ | |
e42eb508 | 72 | // CDROMMethod::GetID - Search the database for a matching string /*{{{*/ |
f46e7681 | 73 | // --------------------------------------------------------------------- |
e42eb508 | 74 | /* */ |
f46e7681 AL |
75 | string CDROMMethod::GetID(string Name) |
76 | { | |
e42eb508 | 77 | // Search for an ID |
f631d1ba | 78 | const Configuration::Item *Top = Database.Tree("CD"); |
b7d9b68e AL |
79 | if (Top != 0) |
80 | Top = Top->Child; | |
e42eb508 | 81 | |
f46e7681 | 82 | for (; Top != 0;) |
e42eb508 | 83 | { |
f46e7681 AL |
84 | if (Top->Value == Name) |
85 | return Top->Tag; | |
e42eb508 | 86 | |
f46e7681 | 87 | Top = Top->Next; |
e42eb508 | 88 | } |
f46e7681 AL |
89 | return string(); |
90 | } | |
91 | /*}}}*/ | |
76fe5db7 MV |
92 | // CDROMMethod::AutoDetectAndMount /*{{{*/ |
93 | // --------------------------------------------------------------------- | |
94 | /* Modifies class varaiable CDROM to the mountpoint */ | |
95 | bool CDROMMethod::AutoDetectAndMount(URI Get) | |
96 | { | |
97 | vector<struct CdromDevice> v = UdevCdroms.Scan(); | |
98 | ||
99 | // first check if its mounted somewhere already | |
100 | for (unsigned int i=0; i < v.size(); i++) | |
101 | { | |
102 | if (v[i].Mounted) | |
103 | { | |
104 | if (Debug) | |
105 | clog << "Checking mounted cdrom device " << v[i].DeviceName << endl; | |
106 | if (IsCorrectCD(Get, v[i].MountPath)) | |
107 | { | |
108 | CDROM = v[i].MountPath; | |
109 | return true; | |
110 | } | |
111 | } | |
112 | } | |
113 | ||
114 | // we are not supposed to mount, exit | |
115 | if (_config->FindB("APT::CDROM::NoMount",false) == true) | |
116 | return false; | |
117 | ||
118 | // check if we have the mount point | |
119 | if (!FileExists("/media/apt")) | |
120 | mkdir("/media/apt", 0755); | |
a6418a4b | 121 | |
76fe5db7 MV |
122 | // now try mounting |
123 | for (unsigned int i=0; i < v.size(); i++) | |
124 | { | |
125 | if (!v[i].Mounted) | |
126 | { | |
127 | if(MountCdrom("/media/apt", v[i].DeviceName)) | |
128 | { | |
129 | if (IsCorrectCD(Get, "/media/apt")) | |
130 | { | |
131 | MountedByApt = true; | |
132 | CDROM = "/media/apt"; | |
133 | return true; | |
134 | } else { | |
135 | UnmountCdrom("/media/apt"); | |
136 | } | |
137 | } | |
138 | } | |
139 | } | |
140 | ||
141 | return false; | |
142 | } | |
143 | /*}}}*/ | |
a6418a4b MV |
144 | // CDROMMethod::IsCorrectCD /*{{{*/ |
145 | // --------------------------------------------------------------------- | |
146 | /* */ | |
147 | bool CDROMMethod::IsCorrectCD(URI want, string MountPath) | |
148 | { | |
a6418a4b MV |
149 | string NewID; |
150 | ||
151 | for (unsigned int Version = 2; Version != 0; Version--) | |
152 | { | |
153 | if (IdentCdrom(MountPath,NewID,Version) == false) | |
154 | return false; | |
155 | ||
76fe5db7 | 156 | if (Debug) |
a6418a4b MV |
157 | clog << "ID " << Version << " " << NewID << endl; |
158 | ||
159 | // A hit | |
160 | if (Database.Find("CD::" + NewID) == want.Host) | |
161 | return true; | |
162 | } | |
163 | ||
164 | return false; | |
165 | } | |
166 | /*}}}*/ | |
f46e7681 AL |
167 | // CDROMMethod::Fetch - Fetch a file /*{{{*/ |
168 | // --------------------------------------------------------------------- | |
169 | /* */ | |
170 | bool CDROMMethod::Fetch(FetchItem *Itm) | |
171 | { | |
76fe5db7 MV |
172 | FetchResult Res; |
173 | ||
f46e7681 AL |
174 | URI Get = Itm->Uri; |
175 | string File = Get.Path; | |
76fe5db7 | 176 | Debug = _config->FindB("Debug::Acquire::cdrom", false); |
34fc0421 | 177 | |
49cb36fc MV |
178 | if (Debug) |
179 | clog << "CDROMMethod::Fetch " << Itm->Uri << endl; | |
34fc0421 | 180 | |
f46e7681 AL |
181 | /* All IMS queries are returned as a hit, CDROMs are readonly so |
182 | time stamps never change */ | |
183 | if (Itm->LastModified != 0) | |
184 | { | |
185 | Res.LastModified = Itm->LastModified; | |
186 | Res.IMSHit = true; | |
2aab5956 | 187 | Res.Filename = Itm->DestFile; |
f46e7681 AL |
188 | URIDone(Res); |
189 | return true; | |
190 | } | |
e42eb508 AL |
191 | |
192 | // Load the database | |
193 | if (DatabaseLoaded == false) | |
194 | { | |
195 | // Read the database | |
196 | string DFile = _config->FindFile("Dir::State::cdroms"); | |
197 | if (FileExists(DFile) == true) | |
198 | { | |
199 | if (ReadConfigFile(Database,DFile) == false) | |
dc738e7a | 200 | return _error->Error(_("Unable to read the cdrom database %s"), |
e42eb508 AL |
201 | DFile.c_str()); |
202 | } | |
203 | DatabaseLoaded = true; | |
204 | } | |
205 | ||
f46e7681 | 206 | // All non IMS queries for package files fail. |
e42eb508 | 207 | if (Itm->IndexFile == true || GetID(Get.Host).empty() == true) |
f46e7681 | 208 | { |
db0db9fe CP |
209 | Fail(_("Please use apt-cdrom to make this CD-ROM recognized by APT." |
210 | " apt-get update cannot be used to add new CD-ROMs")); | |
9e0349cc | 211 | return true; |
f46e7681 AL |
212 | } |
213 | ||
214 | // We already have a CD inserted, but it is the wrong one | |
49cb36fc MV |
215 | if (CurrentID.empty() == false && |
216 | CurrentID != "FAIL" && | |
217 | Database.Find("CD::" + CurrentID) != Get.Host) | |
f46e7681 | 218 | { |
db0db9fe | 219 | Fail(_("Wrong CD-ROM"),true); |
9e0349cc | 220 | return true; |
f46e7681 | 221 | } |
a6418a4b | 222 | |
8e5fc8f5 | 223 | CDROM = _config->FindDir("Acquire::cdrom::mount","/cdrom/"); |
49cb36fc MV |
224 | if (Debug) |
225 | clog << "Looking for CDROM at " << CDROM << endl; | |
a6418a4b | 226 | |
49cb36fc MV |
227 | if (CDROM[0] == '.') |
228 | CDROM= SafeGetCWD() + '/' + CDROM; | |
229 | string NewID; | |
49cb36fc MV |
230 | |
231 | while (CurrentID.empty() == true) | |
a6418a4b | 232 | { |
49cb36fc | 233 | if (CDROM == "apt-udev-auto/") |
76fe5db7 | 234 | AutoDetectAndMount(Get); |
a6418a4b | 235 | |
70dbf5f8 MV |
236 | if(!IsMounted(CDROM)) |
237 | MountedByApt = MountCdrom(CDROM); | |
34fc0421 | 238 | |
a6418a4b | 239 | if (IsCorrectCD(Get, CDROM)) |
175f08ac AL |
240 | break; |
241 | ||
4df0b629 | 242 | // I suppose this should prompt somehow? |
50959877 MV |
243 | if (_config->FindB("APT::CDROM::NoMount",false) == false && |
244 | UnmountCdrom(CDROM) == false) | |
dc738e7a | 245 | return _error->Error(_("Unable to unmount the CD-ROM in %s, it may still be in use."), |
4df0b629 | 246 | CDROM.c_str()); |
018f1533 AL |
247 | if (MediaFail(Get.Host,CDROM) == false) |
248 | { | |
76d97c26 | 249 | CurrentID = "FAIL"; |
2a749770 | 250 | return _error->Error(_("Disk not found.")); |
018f1533 | 251 | } |
f46e7681 AL |
252 | } |
253 | ||
e42eb508 AL |
254 | // Found a CD |
255 | Res.Filename = CDROM + File; | |
256 | struct stat Buf; | |
257 | if (stat(Res.Filename.c_str(),&Buf) != 0) | |
dc738e7a | 258 | return _error->Error(_("File not found")); |
f46e7681 | 259 | |
281daf46 AL |
260 | if (NewID.empty() == false) |
261 | CurrentID = NewID; | |
e42eb508 | 262 | Res.LastModified = Buf.st_mtime; |
e42eb508 | 263 | Res.Size = Buf.st_size; |
13e8426f MV |
264 | |
265 | Hashes Hash; | |
266 | FileFd Fd(Res.Filename, FileFd::ReadOnly); | |
267 | Hash.AddFD(Fd.Fd(), Fd.Size()); | |
268 | Res.TakeHashes(Hash); | |
269 | ||
e42eb508 AL |
270 | URIDone(Res); |
271 | return true; | |
f46e7681 AL |
272 | } |
273 | /*}}}*/ | |
274 | ||
275 | int main() | |
276 | { | |
b25423f6 MZ |
277 | setlocale(LC_ALL, ""); |
278 | ||
f46e7681 AL |
279 | CDROMMethod Mth; |
280 | return Mth.Run(); | |
281 | } |