我在 Linux 上有一个 Python 和 C 应用程序,它应该在从磁盘读取文件时正确处理 IO 错误。该应用程序的大部分是用 Python 编写的,带有执行 IO 的 C 扩展。正是在这个扩展中检测到 IO 错误。
在我看来,错误出现有两种情况。
stat)看起来比使用 fread 读取的要大。我可以相当轻松地测试和处理案例 1。但是,我也想为案例 2 编写单元测试。但是,我不知道如何为测试触发“假”IO 错误。这可能吗?有没有更好的方法来测试这种错误?
最佳答案
errno(3)仅针对
设置为EIO
EIO Input/output error (POSIX.1)
另外,根据 read(2)对于:
EIO I/O error. This will happen for example when the process is in a background process group, tries to read from its controlling terminal, and either it is ignoring or blocking SIGTTIN or its process group is orphaned. It may also occur when there is a low-level I/O error while reading from a disk or tape.
并根据write(2)对于:
EIO A low-level I/O error occurred while modifying the inode.
因此模拟特定的错误代码可能很困难;请注意还有其他 I/O 系统调用,特别是 writev(2)和(间接)mmap(2) , 但 read(2) 和 write(2) 是最常见的。
另请注意 file systems和 Linux kernel (例如它的 VFS 层)是 caching数据。您可能会更晚或永远不会获得 EIO。参见 sync(2)和 fsync(2)
但是,一般来说,大多数软件不会专门处理 EIO w.r.t.其他错误代码;您可能通过获取另一个错误代码进行了足够的测试,例如
EDQUOT The user's quota of disk blocks on the filesystem containing the file referred to by fd has been exhausted.
因此,您可能会通过限制 disk quotas 来进行足够的测试(参见 quotactl(2) 、 setquota(8) 等...)和文件空间(参见 setrlimit(2) 和 RLIMIT_FSIZE、prlimit(1) 、ulimit 内置于 bash(1) 等...)
如果你真的想专门伪造EIO,你可能会物理损坏设备(或者可能只是在错误的时刻拔下 USB 磁盘)或编写你自己的 Filesystem in User Space (FUSE) 模拟它。我认为这不值得付出努力(因为当某些东西得到 EIO 时,整个计算机很快就会变得无法使用,无论如何用户都会注意到......而且因为大多数软件都会处理所有错误代码同样 - 除了 EINTR)
在代码的 C 部分,您可能需要使用 strerror(3) (可能有 syslog(3))和/或 perror(3) .我不确定以与大多数其他错误截然不同的方式处理 EIO 是否值得。
注意:许多关键领域都有定义错误处理方式以及代码开发和测试方式的标准,例如ISO26262在汽车或DO-178B在航空电子设备中。遵循您所在域的标准。
关于python - 在 Linux 上伪造 IO 错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21033660/
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje
我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test
我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c
我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file
我克隆了一个rails仓库,我现在正尝试捆绑安装背景:OSXElCapitanruby2.2.3p173(2015-08-18修订版51636)[x86_64-darwin15]rails-v在您的Gemfile中列出的或native可用的任何gem源中找不到gem'pg(>=0)ruby'。运行bundleinstall以安装缺少的gem。bundleinstallFetchinggemmetadatafromhttps://rubygems.org/............Fetchingversionmetadatafromhttps://rubygems.org/...Fe
在Cooper的书BeginningRuby中,第166页有一个我无法重现的示例。classSongincludeComparableattr_accessor:lengthdef(other)@lengthother.lengthenddefinitialize(song_name,length)@song_name=song_name@length=lengthendenda=Song.new('Rockaroundtheclock',143)b=Song.new('BohemianRhapsody',544)c=Song.new('MinuteWaltz',60)a.betwee
这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下
我是Google云的新手,我正在尝试对其进行首次部署。我的第一个部署是RubyonRails项目。我基本上是在关注thisguideinthegoogleclouddocumentation.唯一的区别是我使用的是我自己的项目,而不是他们提供的“helloworld”项目。这是我的app.yaml文件runtime:customvm:trueentrypoint:bundleexecrackup-p8080-Eproductionconfig.ruresources:cpu:0.5memory_gb:1.3disk_size_gb:10当我转到我的项目目录并运行gcloudprevie
我有两个Rails模型,即Invoice和Invoice_details。一个Invoice_details属于Invoice,一个Invoice有多个Invoice_details。我无法使用accepts_nested_attributes_forinInvoice通过Invoice模型保存Invoice_details。我收到以下错误:(0.2ms)BEGIN(0.2ms)ROLLBACKCompleted422UnprocessableEntityin25ms(ActiveRecord:4.0ms)ActiveRecord::RecordInvalid(Validationfa