Browse Source

Moved samples out so they don't get packaged. Updated test location information. Commented out some debugging as it is a bit overboard now that the original issue is fixed. Modified the gitignore to avoid rolling log files.

master
Drew Short 11 years ago
parent
commit
67aee43791
  1. 6
      .gitignore
  2. BIN
      sample/sample_01_large.jpg
  3. BIN
      sample/sample_01_medium.jpg
  4. BIN
      sample/sample_01_small.jpg
  5. 3
      src/main/resources/default.properties
  6. 3
      src/main/scala/com/sothr/imagetools/hash/DHash.scala
  7. 23
      src/main/scala/com/sothr/imagetools/hash/HashService.scala
  8. 8
      src/main/scala/com/sothr/imagetools/image/ImageService.scala
  9. 3
      src/main/scala/com/sothr/imagetools/util/PropertiesEnum.scala
  10. 3
      src/test/resources/default.properties
  11. 6
      src/test/scala/com/sothr/imagetools/TestParams.scala
  12. 41
      src/test/scala/com/sothr/imagetools/hash/HashServiceTest.scala

6
.gitignore

@ -29,6 +29,6 @@ name.info
version.info version.info
# Image Tools Log Files # Image Tools Log Files
*.debug
*.info
*.err
*.debug*
*.info*
*.err*

BIN
sample/sample_01_large.jpg

After

Width: 3648  |  Height: 2736  |  Size: 5.0 MiB

BIN
sample/sample_01_medium.jpg

After

Width: 1824  |  Height: 1368  |  Size: 1.5 MiB

BIN
sample/sample_01_small.jpg

After

Width: 912  |  Height: 684  |  Size: 519 KiB

3
src/main/resources/default.properties

@ -16,13 +16,16 @@ image.hash.precision=64
image.ahash.use=true image.ahash.use=true
image.ahash.weight=0.70 image.ahash.weight=0.70
image.ahash.precision=8 image.ahash.precision=8
image.ahash.tolerence=8
image.dhash.use=true image.dhash.use=true
image.dhash.weight=0.85 image.dhash.weight=0.85
image.dhash.precision=8 image.dhash.precision=8
image.dhash.tolerence=8
#set to false if hashing images is taking too long #set to false if hashing images is taking too long
image.phash.use=true image.phash.use=true
image.phash.weight=1.0 image.phash.weight=1.0
image.phash.precision=16 image.phash.precision=16
image.phash.tolerence=8
#Default Thumbnail Settings #Default Thumbnail Settings
#Directory where to store thumbnails #Directory where to store thumbnails

3
src/main/scala/com/sothr/imagetools/hash/DHash.scala

@ -13,7 +13,6 @@ object DHash extends PerceptualHasher with Logging {
debug(s"Image data size: ${width}x${height}") debug(s"Image data size: ${width}x${height}")
var hash = 0L var hash = 0L
for (row <- 0 until width) { for (row <- 0 until width) {
//println(f"Row: $row%d")
var previousPixel = imageData(row)(0) var previousPixel = imageData(row)(0)
var previousLocation = (row, 0) var previousLocation = (row, 0)
@ -26,7 +25,7 @@ object DHash extends PerceptualHasher with Logging {
//binary or the current bit based on whether the value //binary or the current bit based on whether the value
//of the current pixel is greater or equal to the previous pixel //of the current pixel is greater or equal to the previous pixel
if (pixel >= previousPixel) hash |= 1 else hash |= 0 if (pixel >= previousPixel) hash |= 1 else hash |= 0
debug(s"hash: hash")
//debug(s"hash: $hash")
previousPixel = pixel previousPixel = pixel
previousLocation = (row, col) previousLocation = (row, col)
} }

23
src/main/scala/com/sothr/imagetools/hash/HashService.scala

@ -2,7 +2,7 @@ package com.sothr.imagetools.hash
import grizzled.slf4j.Logging import grizzled.slf4j.Logging
import com.sothr.imagetools.dto.ImageHashDTO import com.sothr.imagetools.dto.ImageHashDTO
import com.sothr.imagetools.util.{PropertiesEnum, PropertiesService}
import com.sothr.imagetools.util.{PropertiesEnum, PropertiesService, Hamming}
import com.sothr.imagetools.ImageService import com.sothr.imagetools.ImageService
import java.awt.image.BufferedImage import java.awt.image.BufferedImage
import javax.imageio.ImageIO import javax.imageio.ImageIO
@ -66,4 +66,25 @@ object HashService extends Logging {
PHash.getHash(imageData) PHash.getHash(imageData)
} }
def areAhashSimilar(ahash1:Long, ahash2:Long):Boolean = {
val tolerence = PropertiesService.get(PropertiesEnum.AhashTolerence.toString).toInt
val distance = Hamming.getDistance(ahash1, ahash2)
debug(s"hash1: $ahash1 hash2: $ahash2 tolerence: $tolerence hamming distance: $distance")
if (distance <= tolerence) true else false
}
def areDhashSimilar(dhash1:Long, dhash2:Long):Boolean = {
val tolerence = PropertiesService.get(PropertiesEnum.DhashTolerence.toString).toInt
val distance = Hamming.getDistance(dhash1, dhash2)
debug(s"hash1: $dhash1 hash2: $dhash2 tolerence: $tolerence hamming distance: $distance")
if (distance <= tolerence) true else false
}
def arePhashSimilar(phash1:Long, phash2:Long):Boolean = {
val tolerence = PropertiesService.get(PropertiesEnum.PhashTolerence.toString).toInt
val distance = Hamming.getDistance(phash1, phash2)
debug(s"hash1: $phash1 hash2: $phash2 tolerence: $tolerence hamming distance: $distance")
if (distance <= tolerence) true else false
}
} }

8
src/main/scala/com/sothr/imagetools/image/ImageService.scala

@ -70,9 +70,9 @@ object ImageService extends Logging {
var col = 0 var col = 0
debug(s"Processing pixels 0 until $numPixels by $pixelLength") debug(s"Processing pixels 0 until $numPixels by $pixelLength")
for (pixel <- 0 until numPixels by pixelLength) { for (pixel <- 0 until numPixels by pixelLength) {
debug(s"Processing pixel: $pixel/${numPixels - 1}")
//debug(s"Processing pixel: $pixel/${numPixels - 1}")
val argb:Int = pixels(pixel).toInt //singleChannel val argb:Int = pixels(pixel).toInt //singleChannel
debug(s"Pixel data: $argb")
//debug(s"Pixel data: $argb")
result(row)(col) = argb result(row)(col) = argb
col += 1 col += 1
if (col == width) { if (col == width) {
@ -88,7 +88,7 @@ object ImageService extends Logging {
var col = 0 var col = 0
debug(s"Processing pixels 0 until $numPixels by $pixelLength") debug(s"Processing pixels 0 until $numPixels by $pixelLength")
for (pixel <- 0 until numPixels by pixelLength) { for (pixel <- 0 until numPixels by pixelLength) {
debug(s"Processing pixel: $pixel/${numPixels - 1}")
//debug(s"Processing pixel: $pixel/${numPixels - 1}")
var argb:Int = 0 var argb:Int = 0
argb += pixels(pixel).toInt << 24 //alpha argb += pixels(pixel).toInt << 24 //alpha
argb += pixels(pixel + 1).toInt //blue argb += pixels(pixel + 1).toInt //blue
@ -108,7 +108,7 @@ object ImageService extends Logging {
var col = 0 var col = 0
debug(s"Processing pixels 0 until $numPixels by $pixelLength") debug(s"Processing pixels 0 until $numPixels by $pixelLength")
for (pixel <- 0 until numPixels by pixelLength) { for (pixel <- 0 until numPixels by pixelLength) {
debug(s"Processing pixel: $pixel/${numPixels - 1}")
//debug(s"Processing pixel: $pixel/${numPixels - 1}")
var argb:Int = 0 var argb:Int = 0
argb += -16777216; // 255 alpha argb += -16777216; // 255 alpha
argb += pixels(pixel).toInt //blue argb += pixels(pixel).toInt //blue

3
src/main/scala/com/sothr/imagetools/util/PropertiesEnum.scala

@ -14,12 +14,15 @@ object PropertiesEnum extends Enumeration {
val UseAhash = Value("image.ahash.use") val UseAhash = Value("image.ahash.use")
val AhashWeight = Value("image.ahash.weight") val AhashWeight = Value("image.ahash.weight")
val AhashPrecision = Value("image.ahash.precision") val AhashPrecision = Value("image.ahash.precision")
val AhashTolerence = Value("image.ahash.tolerence")
val UseDhash = Value("image.dhash.use") val UseDhash = Value("image.dhash.use")
val DhashWeight = Value("image.dhash.weight") val DhashWeight = Value("image.dhash.weight")
val DhashPrecision = Value("image.dhash.precision") val DhashPrecision = Value("image.dhash.precision")
val DhashTolerence = Value("image.dhash.tolerence")
val UsePhash = Value("image.phash.use") val UsePhash = Value("image.phash.use")
val PhashWeight = Value("image.phash.weight") val PhashWeight = Value("image.phash.weight")
val PhashPrecision = Value("image.phash.precision") val PhashPrecision = Value("image.phash.precision")
val PhashTolerence = Value("image.phash.tolerence")
//Default Thumbnail Settings //Default Thumbnail Settings
val ThumbnailDirectory = Value("thumbnail.directory") val ThumbnailDirectory = Value("thumbnail.directory")
val ThumbnailSize = Value("thumbnail.size") val ThumbnailSize = Value("thumbnail.size")

3
src/test/resources/default.properties

@ -16,13 +16,16 @@ image.hash.precision=64
image.ahash.use=true image.ahash.use=true
image.ahash.weight=0.70 image.ahash.weight=0.70
image.ahash.precision=8 image.ahash.precision=8
image.ahash.tolerence=8
image.dhash.use=true image.dhash.use=true
image.dhash.weight=0.85 image.dhash.weight=0.85
image.dhash.precision=8 image.dhash.precision=8
image.dhash.tolerence=8
#set to false if hashing images is taking too long #set to false if hashing images is taking too long
image.phash.use=true image.phash.use=true
image.phash.weight=1.0 image.phash.weight=1.0
image.phash.precision=16 image.phash.precision=16
image.phash.tolerence=8
#Default Thumbnail Settings #Default Thumbnail Settings
#Directory where to store thumbnails #Directory where to store thumbnails

6
src/test/scala/com/sothr/imagetools/TestParams.scala

@ -1,7 +1,7 @@
package com.sothr.imagetools package com.sothr.imagetools
object TestParams { object TestParams {
val LargeSampleImage1 = "target/sample/sample_01_large.jpg"
val MediumSampleImage1 = "target/sample/sample_01_medium.jpg"
val SmallSampleImage1 = "target/sample/sample_01_small.jpg"
val LargeSampleImage1 = "sample/sample_01_large.jpg"
val MediumSampleImage1 = "sample/sample_01_medium.jpg"
val SmallSampleImage1 = "sample/sample_01_small.jpg"
} }

41
src/test/scala/com/sothr/imagetools/hash/HashServiceTest.scala

@ -9,7 +9,7 @@ import java.io.File
*/ */
class HashServiceTest extends BaseTest { class HashServiceTest extends BaseTest {
def benchmarkDHashTestCase(filePath:String):Long = {
def dhashTestCase(filePath:String):Long = {
val sample = new File(filePath) val sample = new File(filePath)
val image = ImageIO.read(sample) val image = ImageIO.read(sample)
HashService.getDhash(image) HashService.getDhash(image)
@ -18,27 +18,27 @@ class HashServiceTest extends BaseTest {
test("Benchmark DHash") { test("Benchmark DHash") {
info("Benchmarking DHash") info("Benchmarking DHash")
info("DHash Large Image 3684x2736") info("DHash Large Image 3684x2736")
val largeTime1 = getTime { benchmarkDHashTestCase(TestParams.LargeSampleImage1) }
val largeTime2 = getTime { benchmarkDHashTestCase(TestParams.LargeSampleImage1) }
val largeTime3 = getTime { benchmarkDHashTestCase(TestParams.LargeSampleImage1) }
val largeTime4 = getTime { benchmarkDHashTestCase(TestParams.LargeSampleImage1) }
val largeTime5 = getTime { benchmarkDHashTestCase(TestParams.LargeSampleImage1) }
val largeTime1 = getTime { dhashTestCase(TestParams.LargeSampleImage1) }
val largeTime2 = getTime { dhashTestCase(TestParams.LargeSampleImage1) }
val largeTime3 = getTime { dhashTestCase(TestParams.LargeSampleImage1) }
val largeTime4 = getTime { dhashTestCase(TestParams.LargeSampleImage1) }
val largeTime5 = getTime { dhashTestCase(TestParams.LargeSampleImage1) }
val largeMean = getMean(largeTime1, largeTime2, largeTime3, largeTime4, largeTime5) val largeMean = getMean(largeTime1, largeTime2, largeTime3, largeTime4, largeTime5)
info(s"The mean time of 5 tests for large was: $largeMean ms") info(s"The mean time of 5 tests for large was: $largeMean ms")
info("DHash Medium Image 1824x1368") info("DHash Medium Image 1824x1368")
val mediumTime1 = getTime { benchmarkDHashTestCase(TestParams.MediumSampleImage1) }
val mediumTime2 = getTime { benchmarkDHashTestCase(TestParams.MediumSampleImage1) }
val mediumTime3 = getTime { benchmarkDHashTestCase(TestParams.MediumSampleImage1) }
val mediumTime4 = getTime { benchmarkDHashTestCase(TestParams.MediumSampleImage1) }
val mediumTime5 = getTime { benchmarkDHashTestCase(TestParams.MediumSampleImage1) }
val mediumTime1 = getTime { dhashTestCase(TestParams.MediumSampleImage1) }
val mediumTime2 = getTime { dhashTestCase(TestParams.MediumSampleImage1) }
val mediumTime3 = getTime { dhashTestCase(TestParams.MediumSampleImage1) }
val mediumTime4 = getTime { dhashTestCase(TestParams.MediumSampleImage1) }
val mediumTime5 = getTime { dhashTestCase(TestParams.MediumSampleImage1) }
val mediumMean = getMean(mediumTime1, mediumTime2, mediumTime3, mediumTime4, mediumTime5) val mediumMean = getMean(mediumTime1, mediumTime2, mediumTime3, mediumTime4, mediumTime5)
info(s"The mean time of 5 tests for medium was: $mediumMean ms") info(s"The mean time of 5 tests for medium was: $mediumMean ms")
info("DHash Small Image 912x684") info("DHash Small Image 912x684")
val smallTime1 = getTime { benchmarkDHashTestCase(TestParams.SmallSampleImage1) }
val smallTime2 = getTime { benchmarkDHashTestCase(TestParams.SmallSampleImage1) }
val smallTime3 = getTime { benchmarkDHashTestCase(TestParams.SmallSampleImage1) }
val smallTime4 = getTime { benchmarkDHashTestCase(TestParams.SmallSampleImage1) }
val smallTime5 = getTime { benchmarkDHashTestCase(TestParams.SmallSampleImage1) }
val smallTime1 = getTime { dhashTestCase(TestParams.SmallSampleImage1) }
val smallTime2 = getTime { dhashTestCase(TestParams.SmallSampleImage1) }
val smallTime3 = getTime { dhashTestCase(TestParams.SmallSampleImage1) }
val smallTime4 = getTime { dhashTestCase(TestParams.SmallSampleImage1) }
val smallTime5 = getTime { dhashTestCase(TestParams.SmallSampleImage1) }
val smallMean = getMean(smallTime1, smallTime2, smallTime3, smallTime4, smallTime5) val smallMean = getMean(smallTime1, smallTime2, smallTime3, smallTime4, smallTime5)
info(s"The mean time of 5 tests for small was: $smallMean ms") info(s"The mean time of 5 tests for small was: $smallMean ms")
assert(true) assert(true)
@ -77,4 +77,13 @@ class HashServiceTest extends BaseTest {
assert(hash == -5198299688551933030L) assert(hash == -5198299688551933030L)
} }
test("DHash Of Large, Medium, And Small Sample 1 Must Be Similar") {
val largeHash = dhashTestCase(TestParams.LargeSampleImage1)
val mediumHash = dhashTestCase(TestParams.MediumSampleImage1)
val smallHash = dhashTestCase(TestParams.SmallSampleImage1)
assert(HashService.areDhashSimilar(largeHash,mediumHash))
assert(HashService.areDhashSimilar(largeHash,smallHash))
assert(HashService.areDhashSimilar(mediumHash,smallHash))
}
} }
Loading…
Cancel
Save