Skip to content

Content of file MusicBrainzAudioMedatataImporter.java.html

<?xml version="1.0" encoding="UTF-8"?><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"><html xmlns="http://www.w3.org/1999/xhtml" lang="en"><head><meta http-equiv="Content-Type" content="text/html;charset=UTF-8"/><link rel="stylesheet" href="../jacoco-resources/report.css" type="text/css"/><link rel="shortcut icon" href="../jacoco-resources/report.gif" type="image/gif"/><title>MusicBrainzAudioMedatataImporter.java</title><link rel="stylesheet" href="../jacoco-resources/prettify.css" type="text/css"/><script type="text/javascript" src="../jacoco-resources/prettify.js"></script></head><body onload="window['PR_TAB_WIDTH']=4;prettyPrint()"><div class="breadcrumb" id="breadcrumb"><span class="info"><a href="../jacoco-sessions.html" class="el_session">Sessions</a></span><a href="../index.html" class="el_report">blueMarine II :: MusicBrainz</a> &gt; <a href="index.source.html" class="el_package">it.tidalwave.bluemarine2.metadata.impl.audio.musicbrainz</a> &gt; <span class="el_source">MusicBrainzAudioMedatataImporter.java</span></div><h1>MusicBrainzAudioMedatataImporter.java</h1><pre class="source lang-java linenums">/*
 * *********************************************************************************************************************
 *
 * blueMarine II: Semantic Media Centre
 * http://tidalwave.it/projects/bluemarine2
 *
 * Copyright (C) 2015 - 2021 by Tidalwave s.a.s. (http://tidalwave.it)
 *
 * *********************************************************************************************************************
 *
 * Licensed under the Apache License, Version 2.0 (the &quot;License&quot;); you may not use this file except in compliance with
 * the License. You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
 * an &quot;AS IS&quot; BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations under the License.
 *
 * *********************************************************************************************************************
 *
 * git clone https://bitbucket.org/tidalwave/bluemarine2-src
 * git clone https://github.com/tidalwave-it/bluemarine2-src
 *
 * *********************************************************************************************************************
 */
package it.tidalwave.bluemarine2.metadata.impl.audio.musicbrainz;

import javax.annotation.Nonnegative;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.TreeMap;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.stream.Stream;
import java.io.IOException;
import java.math.BigInteger;
import javax.xml.namespace.QName;
import org.apache.commons.lang3.StringUtils;
import org.eclipse.rdf4j.model.IRI;
import org.eclipse.rdf4j.model.Model;
import org.eclipse.rdf4j.model.ValueFactory;
import org.eclipse.rdf4j.model.impl.SimpleValueFactory;
import org.eclipse.rdf4j.model.vocabulary.DC;
import org.eclipse.rdf4j.model.vocabulary.FOAF;
import org.eclipse.rdf4j.model.vocabulary.RDF;
import org.eclipse.rdf4j.model.vocabulary.RDFS;
import org.musicbrainz.ns.mmd_2.Artist;
import org.musicbrainz.ns.mmd_2.DefTrackData;
import org.musicbrainz.ns.mmd_2.Disc;
import org.musicbrainz.ns.mmd_2.Medium;
import org.musicbrainz.ns.mmd_2.MediumList;
import org.musicbrainz.ns.mmd_2.Offset;
import org.musicbrainz.ns.mmd_2.Recording;
import org.musicbrainz.ns.mmd_2.Relation;
import org.musicbrainz.ns.mmd_2.Relation.AttributeList.Attribute;
import org.musicbrainz.ns.mmd_2.RelationList;
import org.musicbrainz.ns.mmd_2.Release;
import org.musicbrainz.ns.mmd_2.ReleaseGroup;
import org.musicbrainz.ns.mmd_2.ReleaseGroupList;
import org.musicbrainz.ns.mmd_2.ReleaseList;
import it.tidalwave.util.Id;
import it.tidalwave.bluemarine2.util.ModelBuilder;
import it.tidalwave.bluemarine2.model.MediaItem;
import it.tidalwave.bluemarine2.model.MediaItem.Metadata;
import it.tidalwave.bluemarine2.model.vocabulary.*;
import it.tidalwave.bluemarine2.metadata.cddb.CddbAlbum;
import it.tidalwave.bluemarine2.metadata.cddb.CddbMetadataProvider;
import it.tidalwave.bluemarine2.metadata.musicbrainz.MusicBrainzMetadataProvider;
import it.tidalwave.bluemarine2.rest.RestResponse;
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.RequiredArgsConstructor;
import lombok.Setter;
import lombok.With;
import lombok.extern.slf4j.Slf4j;
import static java.util.Collections.*;
import static java.util.Comparator.*;
import static java.util.Map.entry;
import static java.util.stream.Collectors.*;
import static it.tidalwave.util.FunctionalCheckedExceptionWrappers.*;
import static it.tidalwave.bluemarine2.util.RdfUtilities.*;
import static it.tidalwave.bluemarine2.model.MediaItem.Metadata.*;
import static it.tidalwave.bluemarine2.metadata.musicbrainz.MusicBrainzMetadataProvider.*;
import static lombok.AccessLevel.PRIVATE;

/***********************************************************************************************************************
 *
 * @author  Fabrizio Giudici
 *
 **********************************************************************************************************************/
<span class="pc bpc" id="L102" title="1 of 2 branches missed.">@Slf4j</span>
<span class="fc" id="L103">@RequiredArgsConstructor</span>
public class MusicBrainzAudioMedatataImporter
  {
<span class="fc" id="L106">    enum Validation</span>
      {
<span class="fc" id="L108">        TRACK_OFFSETS_MATCH_REQUIRED,</span>
<span class="fc" id="L109">        TRACK_OFFSETS_MATCH_NOT_REQUIRED</span>
      }

<span class="fc" id="L112">    private static final QName QNAME_SCORE = new QName(&quot;http://musicbrainz.org/ns/ext#-2.0&quot;, &quot;score&quot;);</span>

<span class="fc" id="L114">    private static final ValueFactory FACTORY = SimpleValueFactory.getInstance();</span>

<span class="fc" id="L116">    private static final String[] TOC_INCLUDES = { &quot;aliases&quot;, &quot;artist-credits&quot;, &quot;labels&quot;, &quot;recordings&quot; };</span>

<span class="fc" id="L118">    private static final String[] RELEASE_INCLUDES = { &quot;aliases&quot;, &quot;artist-credits&quot;, &quot;discids&quot;, &quot;labels&quot;, &quot;recordings&quot; };</span>

<span class="fc" id="L120">    private static final String[] RECORDING_INCLUDES = { &quot;aliases&quot;, &quot;artist-credits&quot;, &quot;artist-rels&quot; };</span>

<span class="fc" id="L122">    private static final IRI SOURCE_MUSICBRAINZ = FACTORY.createIRI(BMMO.NS, &quot;source#musicbrainz&quot;);</span>

    @Nonnull
    private final CddbMetadataProvider cddbMetadataProvider;

    @Nonnull
    private final MusicBrainzMetadataProvider mbMetadataProvider;

<span class="pc" id="L130">    @Getter @Setter</span>
    private int trackOffsetsMatchThreshold = 2500;

<span class="pc" id="L133">    @Getter @Setter</span>
    private int releaseGroupScoreThreshold = 50;

    /** If {@code true}, in case of multiple collections to pick from, those that are not the least one are marked as
        alternative. */
<span class="pc" id="L138">    @Getter @Setter</span>
    private boolean discourageCollections = true;

<span class="fc" id="L141">    private final Set&lt;String&gt; processedTocs = new HashSet&lt;&gt;();</span>

<span class="fc" id="L143">    private static final Map&lt;String, IRI&gt; PERFORMER_MAP = Map.ofEntries(</span>
<span class="fc" id="L144">        entry(&quot;arranger&quot;,                           BMMO.P_ARRANGER),</span>
<span class="fc" id="L145">        entry(&quot;balance&quot;,                            BMMO.P_BALANCE),</span>
<span class="fc" id="L146">        entry(&quot;chorus master&quot;,                      BMMO.P_CHORUS_MASTER),</span>
<span class="fc" id="L147">        entry(&quot;conductor&quot;,                          MO.P_CONDUCTOR),</span>
<span class="fc" id="L148">        entry(&quot;editor&quot;,                             BMMO.P_EDITOR),</span>
<span class="fc" id="L149">        entry(&quot;engineer&quot;,                           MO.P_ENGINEER),</span>
<span class="fc" id="L150">        entry(&quot;instrument arranger&quot;,                BMMO.P_ARRANGER),</span>
<span class="fc" id="L151">        entry(&quot;mastering&quot;,                          BMMO.P_MASTERING),</span>
<span class="fc" id="L152">        entry(&quot;mix&quot;,                                BMMO.P_MIX),</span>
<span class="fc" id="L153">        entry(&quot;orchestrator&quot;,                       BMMO.P_ORCHESTRATOR),</span>
<span class="fc" id="L154">        entry(&quot;performer&quot;,                          MO.P_PERFORMER),</span>
<span class="fc" id="L155">        entry(&quot;performing orchestra&quot;,               BMMO.P_ORCHESTRA),</span>
<span class="fc" id="L156">        entry(&quot;producer&quot;,                           MO.P_PRODUCER),</span>
<span class="fc" id="L157">        entry(&quot;programming&quot;,                        BMMO.P_PROGRAMMING),</span>
<span class="fc" id="L158">        entry(&quot;recording&quot;,                          BMMO.P_RECORDING),</span>
<span class="fc" id="L159">        entry(&quot;remixer&quot;,                            BMMO.P_MIX),</span>
<span class="fc" id="L160">        entry(&quot;sound&quot;,                              MO.P_ENGINEER),</span>

<span class="fc" id="L162">        entry(&quot;vocal&quot;,                              MO.P_SINGER),</span>
<span class="fc" id="L163">        entry(&quot;vocal/additional&quot;,                   BMMO.P_BACKGROUND_SINGER),</span>
<span class="fc" id="L164">        entry(&quot;vocal/alto vocals&quot;,                  BMMO.P_ALTO),</span>
<span class="fc" id="L165">        entry(&quot;vocal/background vocals&quot;,            BMMO.P_BACKGROUND_SINGER),</span>
<span class="fc" id="L166">        entry(&quot;vocal/baritone vocals&quot;,              BMMO.P_BARITONE),</span>
<span class="fc" id="L167">        entry(&quot;vocal/bass-baritone vocals&quot;,         BMMO.P_BASS_BARITONE),</span>
<span class="fc" id="L168">        entry(&quot;vocal/bass vocals&quot;,                  BMMO.P_BASS),</span>
<span class="fc" id="L169">        entry(&quot;vocal/choir vocals&quot;,                 BMMO.P_CHOIR),</span>
<span class="fc" id="L170">        entry(&quot;vocal/contralto vocals&quot;,             BMMO.P_CONTRALTO),</span>
<span class="fc" id="L171">        entry(&quot;vocal/guest&quot;,                        MO.P_SINGER),</span>
<span class="fc" id="L172">        entry(&quot;vocal/lead vocals&quot;,                  BMMO.P_LEAD_SINGER),</span>
<span class="fc" id="L173">        entry(&quot;vocal/mezzo-soprano vocals&quot;,         BMMO.P_MEZZO_SOPRANO),</span>
<span class="fc" id="L174">        entry(&quot;vocal/other vocals&quot;,                 BMMO.P_BACKGROUND_SINGER),</span>
<span class="fc" id="L175">        entry(&quot;vocal/solo&quot;,                         BMMO.P_LEAD_SINGER),</span>
<span class="fc" id="L176">        entry(&quot;vocal/soprano vocals&quot;,               BMMO.P_SOPRANO),</span>
<span class="fc" id="L177">        entry(&quot;vocal/spoken vocals&quot;,                MO.P_SINGER),</span>
<span class="fc" id="L178">        entry(&quot;vocal/tenor vocals&quot;,                 BMMO.P_TENOR),</span>

<span class="fc" id="L180">        entry(&quot;instrument&quot;,                         MO.P_PERFORMER),</span>
<span class="fc" id="L181">        entry(&quot;instrument/accordion&quot;,               BMMO.P_PERFORMER_ACCORDION),</span>
<span class="fc" id="L182">        entry(&quot;instrument/acoustic guitar&quot;,         BMMO.P_PERFORMER_ACOUSTIC_GUITAR),</span>
<span class="fc" id="L183">        entry(&quot;instrument/acoustic bass guitar&quot;,    BMMO.P_PERFORMER_ACOUSTIC_BASS_GUITAR),</span>
<span class="fc" id="L184">        entry(&quot;instrument/agogô&quot;,                   BMMO.P_PERFORMER_AGOGO),</span>
<span class="fc" id="L185">        entry(&quot;instrument/alto saxophone&quot;,          BMMO.P_PERFORMER_ALTO_SAX),</span>
<span class="fc" id="L186">        entry(&quot;instrument/banjo&quot;,                   BMMO.P_PERFORMER_BANJO),</span>
<span class="fc" id="L187">        entry(&quot;instrument/baritone guitar&quot;,         BMMO.P_PERFORMER_BARITONE_GUITAR),</span>
<span class="fc" id="L188">        entry(&quot;instrument/baritone saxophone&quot;,      BMMO.P_PERFORMER_BARITONE_SAX),</span>
<span class="fc" id="L189">        entry(&quot;instrument/bass&quot;,                    BMMO.P_PERFORMER_BASS),</span>
<span class="fc" id="L190">        entry(&quot;instrument/bass clarinet&quot;,           BMMO.P_PERFORMER_BASS_CLARINET),</span>
<span class="fc" id="L191">        entry(&quot;instrument/bass drum&quot;,               BMMO.P_PERFORMER_BASS_DRUM),</span>
<span class="fc" id="L192">        entry(&quot;instrument/bass guitar&quot;,             BMMO.P_PERFORMER_BASS_GUITAR),</span>
<span class="fc" id="L193">        entry(&quot;instrument/bass trombone&quot;,           BMMO.P_PERFORMER_BASS_TROMBONE),</span>
<span class="fc" id="L194">        entry(&quot;instrument/bassoon&quot;,                 BMMO.P_PERFORMER_BASSOON),</span>
<span class="fc" id="L195">        entry(&quot;instrument/bells&quot;,                   BMMO.P_PERFORMER_BELLS),</span>
<span class="fc" id="L196">        entry(&quot;instrument/berimbau&quot;,                BMMO.P_PERFORMER_BERIMBAU),</span>
<span class="fc" id="L197">        entry(&quot;instrument/brass&quot;,                   BMMO.P_PERFORMER_BRASS),</span>
<span class="fc" id="L198">        entry(&quot;instrument/brushes&quot;,                 BMMO.P_PERFORMER_BRUSHES),</span>
<span class="fc" id="L199">        entry(&quot;instrument/cello&quot;,                   BMMO.P_PERFORMER_CELLO),</span>
<span class="fc" id="L200">        entry(&quot;instrument/clarinet&quot;,                BMMO.P_PERFORMER_CLARINET),</span>
<span class="fc" id="L201">        entry(&quot;instrument/classical guitar&quot;,        BMMO.P_PERFORMER_CLASSICAL_GUITAR),</span>
<span class="fc" id="L202">        entry(&quot;instrument/congas&quot;,                  BMMO.P_PERFORMER_CONGAS),</span>
<span class="fc" id="L203">        entry(&quot;instrument/cornet&quot;,                  BMMO.P_PERFORMER_CORNET),</span>
<span class="fc" id="L204">        entry(&quot;instrument/cymbals&quot;,                 BMMO.P_PERFORMER_CYMBALS),</span>
<span class="fc" id="L205">        entry(&quot;instrument/double bass&quot;,             BMMO.P_PERFORMER_DOUBLE_BASS),</span>
<span class="fc" id="L206">        entry(&quot;instrument/drums&quot;,                   BMMO.P_PERFORMER_DRUMS),</span>
<span class="fc" id="L207">        entry(&quot;instrument/drum machine&quot;,            BMMO.P_PERFORMER_DRUM_MACHINE),</span>
<span class="fc" id="L208">        entry(&quot;instrument/electric bass guitar&quot;,    BMMO.P_PERFORMER_ELECTRIC_BASS_GUITAR),</span>
<span class="fc" id="L209">        entry(&quot;instrument/electric guitar&quot;,         BMMO.P_PERFORMER_ELECTRIC_GUITAR),</span>
<span class="fc" id="L210">        entry(&quot;instrument/electric piano&quot;,          BMMO.P_PERFORMER_ELECTRIC_PIANO),</span>
<span class="fc" id="L211">        entry(&quot;instrument/electric sitar&quot;,          BMMO.P_PERFORMER_ELECTRIC_SITAR),</span>
<span class="fc" id="L212">        entry(&quot;instrument/electronic drum set&quot;,     BMMO.P_PERFORMER_ELECTRONIC_DRUM_SET),</span>
<span class="fc" id="L213">        entry(&quot;instrument/english horn&quot;,            BMMO.P_PERFORMER_ENGLISH_HORN),</span>
<span class="fc" id="L214">        entry(&quot;instrument/flugelhorn&quot;,              BMMO.P_PERFORMER_FLUGELHORN),</span>
<span class="fc" id="L215">        entry(&quot;instrument/flute&quot;,                   BMMO.P_PERFORMER_FLUTE),</span>
<span class="fc" id="L216">        entry(&quot;instrument/frame drum&quot;,              BMMO.P_PERFORMER_FRAME_DRUM),</span>
<span class="fc" id="L217">        entry(&quot;instrument/french horn&quot;,             BMMO.P_PERFORMER_FRENCH_HORN),</span>
<span class="fc" id="L218">        entry(&quot;instrument/glockenspiel&quot;,            BMMO.P_PERFORMER_GLOCKENSPIEL),</span>
<span class="fc" id="L219">        entry(&quot;instrument/grand piano&quot;,             BMMO.P_PERFORMER_GRAND_PIANO),</span>
<span class="fc" id="L220">        entry(&quot;instrument/guest&quot;,                   BMMO.P_PERFORMER_GUEST),</span>
<span class="fc" id="L221">        entry(&quot;instrument/guitar&quot;,                  BMMO.P_PERFORMER_GUITAR),</span>
<span class="fc" id="L222">        entry(&quot;instrument/guitar synthesizer&quot;,      BMMO.P_PERFORMER_GUITAR_SYNTHESIZER),</span>
<span class="fc" id="L223">        entry(&quot;instrument/guitars&quot;,                 BMMO.P_PERFORMER_GUITARS),</span>
<span class="fc" id="L224">        entry(&quot;instrument/handclaps&quot;,               BMMO.P_PERFORMER_HANDCLAPS),</span>
<span class="fc" id="L225">        entry(&quot;instrument/hammond organ&quot;,           BMMO.P_PERFORMER_HAMMOND_ORGAN),</span>
<span class="fc" id="L226">        entry(&quot;instrument/harmonica&quot;,               BMMO.P_PERFORMER_HARMONICA),</span>
<span class="fc" id="L227">        entry(&quot;instrument/harp&quot;,                    BMMO.P_PERFORMER_HARP),</span>
<span class="fc" id="L228">        entry(&quot;instrument/harpsichord&quot;,             BMMO.P_PERFORMER_HARPSICHORD),</span>
<span class="fc" id="L229">        entry(&quot;instrument/hi-hat&quot;,                  BMMO.P_PERFORMER_HIHAT),</span>
<span class="fc" id="L230">        entry(&quot;instrument/horn&quot;,                    BMMO.P_PERFORMER_HORN),</span>
<span class="fc" id="L231">        entry(&quot;instrument/keyboard&quot;,                BMMO.P_PERFORMER_KEYBOARD),</span>
<span class="fc" id="L232">        entry(&quot;instrument/koto&quot;,                    BMMO.P_PERFORMER_KOTO),</span>
<span class="fc" id="L233">        entry(&quot;instrument/lute&quot;,                    BMMO.P_PERFORMER_LUTE),</span>
<span class="fc" id="L234">        entry(&quot;instrument/maracas&quot;,                 BMMO.P_PERFORMER_MARACAS),</span>
<span class="fc" id="L235">        entry(&quot;instrument/marimba&quot;,                 BMMO.P_PERFORMER_MARIMBA),</span>
<span class="fc" id="L236">        entry(&quot;instrument/mellophone&quot;,              BMMO.P_PERFORMER_MELLOPHONE),</span>
<span class="fc" id="L237">        entry(&quot;instrument/melodica&quot;,                BMMO.P_PERFORMER_MELODICA),</span>
<span class="fc" id="L238">        entry(&quot;instrument/oboe&quot;,                    BMMO.P_PERFORMER_OBOE),</span>
<span class="fc" id="L239">        entry(&quot;instrument/organ&quot;,                   BMMO.P_PERFORMER_ORGAN),</span>
<span class="fc" id="L240">        entry(&quot;instrument/other instruments&quot;,       BMMO.P_PERFORMER_OTHER_INSTRUMENTS),</span>
<span class="fc" id="L241">        entry(&quot;instrument/percussion&quot;,              BMMO.P_PERFORMER_PERCUSSION),</span>
<span class="fc" id="L242">        entry(&quot;instrument/piano&quot;,                   BMMO.P_PERFORMER_PIANO),</span>
<span class="fc" id="L243">        entry(&quot;instrument/piccolo trumpet&quot;,         BMMO.P_PERFORMER_PICCOLO_TRUMPET),</span>
<span class="fc" id="L244">        entry(&quot;instrument/pipe organ&quot;,              BMMO.P_PERFORMER_PIPE_ORGAN),</span>
<span class="fc" id="L245">        entry(&quot;instrument/psaltery&quot;,                BMMO.P_PERFORMER_PSALTERY),</span>
<span class="fc" id="L246">        entry(&quot;instrument/recorder&quot;,                BMMO.P_PERFORMER_RECORDER),</span>
<span class="fc" id="L247">        entry(&quot;instrument/reeds&quot;,                   BMMO.P_PERFORMER_REEDS),</span>
<span class="fc" id="L248">        entry(&quot;instrument/rhodes piano&quot;,            BMMO.P_PERFORMER_RHODES_PIANO),</span>
<span class="fc" id="L249">        entry(&quot;instrument/santur&quot;,                  BMMO.P_PERFORMER_SANTUR),</span>
<span class="fc" id="L250">        entry(&quot;instrument/saxophone&quot;,               BMMO.P_PERFORMER_SAXOPHONE),</span>
<span class="fc" id="L251">        entry(&quot;instrument/shakers&quot;,                 BMMO.P_PERFORMER_SHAKERS),</span>
<span class="fc" id="L252">        entry(&quot;instrument/sitar&quot;,                   BMMO.P_PERFORMER_SITAR),</span>
<span class="fc" id="L253">        entry(&quot;instrument/slide guitar&quot;,            BMMO.P_PERFORMER_SLIDE_GUITAR),</span>
<span class="fc" id="L254">        entry(&quot;instrument/snare drum&quot;,              BMMO.P_PERFORMER_SNARE_DRUM),</span>
<span class="fc" id="L255">        entry(&quot;instrument/solo&quot;,                    BMMO.P_PERFORMER_SOLO),</span>
<span class="fc" id="L256">        entry(&quot;instrument/soprano saxophone&quot;,       BMMO.P_PERFORMER_SOPRANO_SAX),</span>
<span class="fc" id="L257">        entry(&quot;instrument/spanish acoustic guitar&quot;, BMMO.P_PERFORMER_SPANISH_ACOUSTIC_GUITAR),</span>
<span class="fc" id="L258">        entry(&quot;instrument/steel guitar&quot;,            BMMO.P_PERFORMER_STEEL_GUITAR),</span>
<span class="fc" id="L259">        entry(&quot;instrument/synclavier&quot;,              BMMO.P_PERFORMER_SYNCLAVIER),</span>
<span class="fc" id="L260">        entry(&quot;instrument/synthesizer&quot;,             BMMO.P_PERFORMER_SYNTHESIZER),</span>
<span class="fc" id="L261">        entry(&quot;instrument/tambourine&quot;,              BMMO.P_PERFORMER_TAMBOURINE),</span>
<span class="fc" id="L262">        entry(&quot;instrument/tenor saxophone&quot;,         BMMO.P_PERFORMER_TENOR_SAX),</span>
<span class="fc" id="L263">        entry(&quot;instrument/timbales&quot;,                BMMO.P_PERFORMER_TIMBALES),</span>
<span class="fc" id="L264">        entry(&quot;instrument/timpani&quot;,                 BMMO.P_PERFORMER_TIMPANI),</span>
<span class="fc" id="L265">        entry(&quot;instrument/tiple&quot;,                   BMMO.P_PERFORMER_TIPLE),</span>
<span class="fc" id="L266">        entry(&quot;instrument/trombone&quot;,                BMMO.P_PERFORMER_TROMBONE),</span>
<span class="fc" id="L267">        entry(&quot;instrument/trumpet&quot;,                 BMMO.P_PERFORMER_TRUMPET),</span>
<span class="fc" id="L268">        entry(&quot;instrument/tuba&quot;,                    BMMO.P_PERFORMER_TUBA),</span>
<span class="fc" id="L269">        entry(&quot;instrument/tubular bells&quot;,           BMMO.P_PERFORMER_TUBULAR_BELLS),</span>
<span class="fc" id="L270">        entry(&quot;instrument/tuned percussion&quot;,        BMMO.P_PERFORMER_TUNED_PERCUSSION),</span>
<span class="fc" id="L271">        entry(&quot;instrument/ukulele&quot;,                 BMMO.P_PERFORMER_UKULELE),</span>
<span class="fc" id="L272">        entry(&quot;instrument/vibraphone&quot;,              BMMO.P_PERFORMER_VIBRAPHONE),</span>
<span class="fc" id="L273">        entry(&quot;instrument/viola&quot;,                   BMMO.P_PERFORMER_VIOLA),</span>
<span class="fc" id="L274">        entry(&quot;instrument/viola da gamba&quot;,          BMMO.P_PERFORMER_VIOLA_DA_GAMBA),</span>
<span class="fc" id="L275">        entry(&quot;instrument/violin&quot;,                  BMMO.P_PERFORMER_VIOLIN),</span>
<span class="fc" id="L276">        entry(&quot;instrument/whistle&quot;,                 BMMO.P_PERFORMER_WHISTLE),</span>
<span class="fc" id="L277">        entry(&quot;instrument/xylophone&quot;,               BMMO.P_PERFORMER_XYLOPHONE));</span>

    /*******************************************************************************************************************
     *
     * Aggregate of a {@link Release}, a {@link Medium} inside that {@code Release} and a {@link Disc} inside that
     * {@code Medium}.
     *
     ******************************************************************************************************************/
<span class="fc" id="L285">    @RequiredArgsConstructor @AllArgsConstructor @Getter</span>
    static class ReleaseMediumDisk
      {
        @Nonnull
<span class="fc" id="L289">        private final Release release;</span>

        @Nonnull
<span class="fc" id="L292">        private final Medium medium;</span>

<span class="pc bpc" id="L294" title="1 of 2 branches missed.">        @With</span>
<span class="fc" id="L295">        private Disc disc;</span>

<span class="fc bfc" id="L297" title="All 2 branches covered.">        @With</span>
<span class="fc" id="L298">        private boolean alternative;</span>

<span class="nc" id="L300">        private String embeddedTitle;</span>

<span class="fc" id="L302">        private int score;</span>

        /***************************************************************************************************************
         *
         **************************************************************************************************************/
        @Nonnull
        public ReleaseMediumDisk withEmbeddedTitle (@Nonnull final String embeddedTitle)
          {
<span class="fc" id="L310">            return new ReleaseMediumDisk(release, medium, disc, alternative, embeddedTitle,</span>
<span class="fc" id="L311">                                         similarity(pickTitle(), embeddedTitle));</span>
          }

        /***************************************************************************************************************
         *
         * Prefer Medium title - typically available in case of disk collections, in which case Release has got
         * the collection title, which is very generic.
         *
         **************************************************************************************************************/
        @Nonnull
        public String pickTitle()
          {
<span class="fc" id="L323">            return Optional.ofNullable(medium.getTitle()).orElse(release.getTitle());</span>
          }

        /***************************************************************************************************************
         *
         **************************************************************************************************************/
        @Nonnull
        public ReleaseMediumDisk alternativeIf (final boolean condition)
          {
<span class="fc bfc" id="L332" title="All 4 branches covered.">            return withAlternative(alternative || condition);</span>
          }

        /***************************************************************************************************************
         *
         **************************************************************************************************************/
        @Nonnull
        public Id computeId()
          {
<span class="fc" id="L341">            return createSha1IdNew(getRelease().getId() + &quot;+&quot; + getDisc().getId());</span>
          }

        /***************************************************************************************************************
         *
         **************************************************************************************************************/
        @Nonnull
        public Optional&lt;Integer&gt; getDiskCount()
          {
<span class="fc" id="L350">            return Optional.ofNullable(release.getMediumList()).map(MediumList::getCount).map(BigInteger::intValue);</span>
          }

        /***************************************************************************************************************
         *
         **************************************************************************************************************/
        @Nonnull
        public Optional&lt;Integer&gt; getDiskNumber()
          {
<span class="fc" id="L359">            return Optional.ofNullable(medium.getPosition()).map(BigInteger::intValue);</span>
          }

        /***************************************************************************************************************
         *
         **************************************************************************************************************/
        @Nonnull
        public Optional&lt;String&gt; getAsin()
          {
<span class="fc" id="L368">            return Optional.ofNullable(release.getAsin());</span>
          }

        /***************************************************************************************************************
         *
         **************************************************************************************************************/
        @Nonnull
        public Optional&lt;String&gt; getBarcode()
          {
<span class="fc" id="L377">            return Optional.ofNullable(release.getBarcode());</span>
          }

        /***************************************************************************************************************
         *
         **************************************************************************************************************/
        @Nonnull
        public Cddb getCddb()
          {
<span class="fc" id="L386">            return MediaItem.Metadata.Cddb.builder()</span>
<span class="fc" id="L387">                    .discId(&quot;&quot;) // FIXME</span>
<span class="fc" id="L388"> .trackFrameOffsets(disc.getOffsetList().getOffset()</span> <span class="fc" id="L389"> .stream()</span> <span class="fc" id="L390"> .map(Offset::getValue)</span> <span class="fc" id="L391"> .mapToInt(BigInteger::intValue)</span> <span class="fc" id="L392"> .toArray())</span> <span class="fc" id="L393"> .build();</span> } /*************************************************************************************************************** * **************************************************************************************************************/ @Nonnull public String getMediumAndDiscString() { <span class="fc bfc" id="L402" title="All 2 branches covered."> return String.format(&quot;%s/%s&quot;, medium.getTitle(), (disc != null) ? disc.getId() : &quot;null&quot;);</span> } /*************************************************************************************************************** * **************************************************************************************************************/ @Override public boolean equals (@Nullable final Object other) { <span class="pc bpc" id="L411" title="1 of 2 branches missed."> if (this == other)</span> { <span class="nc" id="L413"> return true;</span> } <span class="pc bpc" id="L416" title="2 of 4 branches missed."> if ((other == null) || (getClass() != other.getClass()))</span> { <span class="nc" id="L418"> return false;</span> } <span class="fc" id="L421"> return Objects.equals(this.computeId(), ((ReleaseMediumDisk)other).computeId());</span> } /*************************************************************************************************************** * **************************************************************************************************************/ @Override public int hashCode() { <span class="fc" id="L430"> return computeId().hashCode();</span> } /*************************************************************************************************************** * **************************************************************************************************************/ @Override @Nonnull public String toString() { <span class="fc" id="L439"> return String.format(&quot;ALT: %-5s %s ASIN: %-10s BARCODE: %-13s SCORE: %4d #: %3s/%3s &quot; +</span> &quot;TITLES: PICKED: %s EMBEDDED: %s RELEASE: %s MEDIUM: %s&quot;, <span class="fc" id="L441"> alternative,</span> <span class="fc" id="L442"> release.getId(),</span> <span class="fc" id="L443"> release.getAsin(),</span> <span class="fc" id="L444"> release.getBarcode(),</span> <span class="fc" id="L445"> getScore(),</span> <span class="fc" id="L446"> getDiskNumber().map(Number::toString).orElse(&quot;&quot;),</span> <span class="fc" id="L447"> getDiskCount().map(Number::toString).orElse(&quot;&quot;),</span> <span class="fc" id="L448"> pickTitle(), embeddedTitle, release.getTitle(), medium.getTitle());</span> } } /******************************************************************************************************************* * * Aggregate of a {@link Relation} and a target type. * ******************************************************************************************************************/ <span class="fc" id="L457"> @RequiredArgsConstructor(access = PRIVATE) @Getter</span> static class RelationAndTargetType { @Nonnull <span class="fc" id="L461"> private final Relation relation;</span> @Nonnull <span class="fc" id="L464"> private final String targetType;</span> @Nonnull public static Stream&lt;RelationAndTargetType&gt; toStream (@Nonnull final RelationList relationList) { <span class="fc" id="L469"> return relationList.getRelation().stream()</span> <span class="fc" id="L470"> .map(rel -&gt; new RelationAndTargetType(rel, relationList.getTargetType()));</span> } } /******************************************************************************************************************* * * Downloads and imports MusicBrainz data for the given {@link Metadata}. * * @param metadata the {@code Metadata} * @return the RDF triples * @throws InterruptedException in case of I/O error * @throws IOException in case of I/O error * ******************************************************************************************************************/ @Nonnull public Optional&lt;Model&gt; handleMetadata (@Nonnull final Metadata metadata) throws InterruptedException, IOException { <span class="fc" id="L488"> final ModelBuilder model = createModelBuilder();</span> <span class="fc" id="L489"> final Optional&lt;String&gt; optionalAlbumTitle = metadata.get(ALBUM);</span> <span class="fc" id="L490"> final Optional&lt;Cddb&gt; optionalCddb = metadata.get(CDDB);</span> <span class="pc bpc" id="L492" title="2 of 6 branches missed."> if (optionalAlbumTitle.isPresent() &amp;&amp; !optionalAlbumTitle.get().trim().isEmpty() &amp;&amp; optionalCddb.isPresent())</span> { <span class="fc" id="L494"> final String albumTitle = optionalAlbumTitle.get();</span> <span class="fc" id="L495"> final Cddb cddb = optionalCddb.get();</span> <span class="fc" id="L496"> final String toc = cddb.getToc();</span> <span class="fc" id="L498"> synchronized (processedTocs)</span> { <span class="fc bfc" id="L500" title="All 2 branches covered."> if (processedTocs.contains(toc))</span> { <span class="fc" id="L502"> return Optional.empty();</span> } <span class="fc" id="L505"> processedTocs.add(toc);</span> <span class="fc" id="L506"> }</span> <span class="fc" id="L508"> log.info(&quot;QUERYING MUSICBRAINZ FOR TOC OF: {}&quot;, albumTitle);</span> <span class="fc" id="L509"> final List&lt;ReleaseMediumDisk&gt; rmds = new ArrayList&lt;&gt;();</span> <span class="fc" id="L510"> final RestResponse&lt;ReleaseList&gt; releaseList = mbMetadataProvider.findReleaseListByToc(toc, TOC_INCLUDES);</span> // even though we're querying by TOC, matching offsets is required to kill many false results <span class="fc" id="L512"> releaseList.ifPresent(releases -&gt; rmds.addAll(findReleases(releases, cddb, Validation.TRACK_OFFSETS_MATCH_REQUIRED)));</span> <span class="fc bfc" id="L514" title="All 2 branches covered."> if (rmds.isEmpty())</span> { <span class="fc" id="L516"> log.info(&quot;TOC NOT FOUND, QUERYING MUSICBRAINZ FOR TITLE: {}&quot;, albumTitle);</span> <span class="fc" id="L517"> final List&lt;ReleaseGroup&gt; releaseGroups = new ArrayList&lt;&gt;();</span> <span class="fc" id="L518"> releaseGroups.addAll(mbMetadataProvider.findReleaseGroupByTitle(albumTitle)</span> <span class="fc" id="L519"> .map(ReleaseGroupList::getReleaseGroup)</span> <span class="fc" id="L520"> .orElse(emptyList()));</span> <span class="fc" id="L522"> final Optional&lt;String&gt; alternateTitle = cddbAlternateTitleOf(metadata);</span> <span class="fc" id="L523"> alternateTitle.ifPresent(t -&gt; log.info(&quot;ALSO USING ALTERNATE TITLE: {}&quot;, t));</span> <span class="fc" id="L524"> releaseGroups.addAll(alternateTitle.map(_f(mbMetadataProvider::findReleaseGroupByTitle))</span> <span class="fc" id="L525"> .map(response -&gt; response.get().getReleaseGroup())</span> <span class="fc" id="L526"> .orElse(emptyList()));</span> <span class="fc" id="L527"> rmds.addAll(findReleases(releaseGroups, cddb, Validation.TRACK_OFFSETS_MATCH_REQUIRED));</span> } <span class="fc" id="L530"> model.with(markedAlternative(rmds, albumTitle).stream()</span> <span class="fc" id="L531"> .parallel()</span> <span class="fc" id="L532"> .map(_f(rmd -&gt; handleRelease(metadata, rmd)))</span> <span class="fc" id="L533"> .collect(toList()));</span> } <span class="fc" id="L536"> return Optional.of(model.toModel());</span> } /******************************************************************************************************************* * * Given a valid list of {@link ReleaseMediumDisk}s - that is, that has been already validated and correctly matches * the searched record - if it contains more than one element picks the most suitable one. Unwanted elements are * not filtered out, because it's not always possible to automatically pick the best one: in fact, some entries * might differ for ASIN or barcode; or might be items individually sold or part of a collection. It makes sense to * offer the user the possibility of manually pick them later. So, instead of being filtered out, those elements * are marked as &quot;alternative&quot; (and they will be later marked as such in the triple store). * * These are the performed steps: * * &lt;ol&gt; * &lt;li&gt;Eventual duplicates are collapsed.&lt;/li&gt; * &lt;li&gt;If required, in case of members of collections, collections that are larger than the least are marked as * alternative.&lt;/li&gt; * &lt;li&gt;A matching score is computed about the affinity of the title found in MusicBrainz metadata with respect * to the title in the embedded metadata: elements that don't reach the maximum score are marked as alternative. * &lt;/li&gt; * &lt;li&gt;If at least one element has got an ASIN, other elements that don't bear it are marked as alternative.&lt;/li&gt; * &lt;li&gt;If at least one element has got a barcode, other elements that don't bear it are marked as alternative.&lt;/li&gt; * &lt;li&gt;If the pick is not unique yet, an ASIN is picked as the first in lexicographic order and elements not * bearing it are marked as alternative.&lt;/li&gt; * &lt;li&gt;If the pick is not unique yet, a barcode is picked as the first in lexicographic order and elements not * bearing it are marked as alternative.&lt;/li&gt; * &lt;li&gt;If the pick is not unique yet, elements other than the first one are marked as alternative.&lt;/i&gt; * &lt;/ol&gt; * * The last criteria are implemented for giving consistency to automated tests, considering that the order in which * elements are found is not guaranteed because of multi-threading. * * @param inRmds the incoming {@code ReleaseAndMedium}s * @param embeddedTitle the album title found in the file * @return the processed {@code ReleaseAndMedium}s * ******************************************************************************************************************/ @Nonnull private List&lt;ReleaseMediumDisk&gt; markedAlternative (@Nonnull final List&lt;ReleaseMediumDisk&gt; inRmds, @Nonnull final String embeddedTitle) { <span class="fc bfc" id="L578" title="All 2 branches covered."> if (inRmds.size() &lt;= 1)</span> { <span class="fc" id="L580"> return inRmds;</span> } <span class="fc" id="L583"> List&lt;ReleaseMediumDisk&gt; rmds = inRmds.stream()</span> <span class="fc" id="L584"> .map(rmd -&gt; rmd.withEmbeddedTitle(embeddedTitle))</span> <span class="fc" id="L585"> .distinct()</span> <span class="fc" id="L586"> .collect(toList());</span> <span class="pc bpc" id="L587" title="1 of 2 branches missed."> rmds = discourageCollections ? markedAlternativeIfNotLeastCollection(rmds) : rmds;</span> <span class="fc" id="L588"> rmds = markedAlternativeByTitleAffinity(rmds);</span> <span class="fc" id="L589"> rmds = markedAlternativeByAsinOrBarcode(rmds);</span> <span class="fc" id="L590"> rmds = markedAlternativeButTheFirstNotAlternative(rmds);</span> <span class="fc" id="L592"> synchronized (log) // keep log lines together</span> { <span class="fc" id="L594"> log.info(&quot;MULTIPLE RESULTS&quot;);</span> <span class="fc" id="L595"> rmds.forEach(rmd -&gt; log.info(&quot;&gt;&gt;&gt; MULTIPLE RESULTS: {}&quot;, rmd));</span> <span class="fc" id="L596"> }</span> <span class="fc" id="L598"> final int count = countOfNotAlternative(rmds);</span> <span class="pc bpc" id="L599" title="2 of 4 branches missed."> assert count == 1 : &quot;Still too many items not alternative: &quot; + count;</span> <span class="fc" id="L601"> return rmds;</span> } /******************************************************************************************************************* * * @param rmds the incoming {@code ReleaseMediumDisk} * @return the processed {@code ReleaseMediumDisk} * ******************************************************************************************************************/ @Nonnull private static List&lt;ReleaseMediumDisk&gt; markedAlternativeByAsinOrBarcode (@Nonnull List&lt;ReleaseMediumDisk&gt; rmds) { <span class="fc bfc" id="L613" title="All 4 branches covered."> final boolean asinPresent = rmds.stream().anyMatch(rmd -&gt; !rmd.isAlternative() &amp;&amp; rmd.getAsin().isPresent());</span> <span class="fc bfc" id="L614" title="All 4 branches covered."> rmds = markedAlternative(rmds, rmd -&gt; asinPresent &amp;&amp; rmd.getAsin().isEmpty());</span> <span class="fc" id="L616"> final boolean barcodePresent =</span> <span class="fc bfc" id="L617" title="All 4 branches covered."> rmds.stream().anyMatch(rmd -&gt; !rmd.isAlternative() &amp;&amp; rmd.getBarcode().isPresent());</span> <span class="fc bfc" id="L618" title="All 4 branches covered."> rmds = markedAlternative(rmds, rmd -&gt; barcodePresent &amp;&amp; rmd.getBarcode().isEmpty());</span> <span class="fc bfc" id="L620" title="All 4 branches covered."> if (asinPresent &amp;&amp; (countOfNotAlternative(rmds) &gt; 1))</span> { <span class="fc" id="L622"> final Optional&lt;String&gt; asin = findFirstNotInAlternative(rmds, rmd -&gt; rmd.getAsin());</span> <span class="fc bfc" id="L623" title="All 2 branches covered."> rmds = markedAlternative(rmds, rmd -&gt; !rmd.getAsin().equals(asin));</span> } <span class="fc bfc" id="L626" title="All 4 branches covered."> if (barcodePresent &amp;&amp; (countOfNotAlternative(rmds) &gt; 1))</span> { <span class="fc" id="L628"> final Optional&lt;String&gt; barcode = findFirstNotInAlternative(rmds, rmd -&gt; rmd.getBarcode());</span> <span class="fc bfc" id="L629" title="All 2 branches covered."> rmds = markedAlternative(rmds, rmd -&gt; !rmd.getBarcode().equals(barcode));</span> } <span class="fc" id="L632"> return rmds;</span> } /******************************************************************************************************************* * * Sweeps the given {@link ReleaseMediumDisk}s and marks as alternative all the items after a not alternative item. * * @param rmds the incoming {@code ReleaseMediumDisk} * @return the processed {@code ReleaseMediumDisk} * ******************************************************************************************************************/ @Nonnull private static List&lt;ReleaseMediumDisk&gt; markedAlternativeButTheFirstNotAlternative (@Nonnull final List&lt;ReleaseMediumDisk&gt; rmds) { <span class="fc bfc" id="L646" title="All 2 branches covered."> if (countOfNotAlternative(rmds) &lt;= 1)</span> { <span class="fc" id="L648"> return rmds;</span> } <span class="fc" id="L651"> final ReleaseMediumDisk pick = rmds.stream()</span> <span class="fc bfc" id="L652" title="All 2 branches covered."> .filter(rmd -&gt; !rmd.isAlternative())</span> <span class="fc" id="L653"> .sorted(comparing(rmd -&gt; rmd.getRelease().getId())) // Fix for BMT-166</span> <span class="fc" id="L654"> .findFirst()</span> <span class="fc" id="L655"> .get();</span> <span class="fc bfc" id="L656" title="All 2 branches covered."> return markedAlternative(rmds, rmd -&gt; rmd != pick);</span> } /******************************************************************************************************************* * * Sweeps the given {@link ReleaseMediumDisk}s and marks as alternative all the items which are not part of the * disk collections with the minimum size. * * @param rmds the incoming {@code ReleaseMediumDisk}s * @return the processed {@code ReleaseMediumDisk}s * ******************************************************************************************************************/ @Nonnull private static List&lt;ReleaseMediumDisk&gt; markedAlternativeIfNotLeastCollection (@Nonnull final List&lt;ReleaseMediumDisk&gt; rmds) { <span class="pc bpc" id="L671" title="1 of 2 branches missed."> final int leastSize = rmds.stream().filter(rmd -&gt; !rmd.isAlternative())</span> <span class="fc" id="L672"> .mapToInt(rmd -&gt; rmd.getDiskCount().orElse(1))</span> <span class="fc" id="L673"> .min().getAsInt();</span> <span class="fc bfc" id="L674" title="All 2 branches covered."> return markedAlternative(rmds, rmd -&gt; rmd.getDiskCount().orElse(1) &gt; leastSize);</span> } /******************************************************************************************************************* * * Sweeps the given {@link ReleaseMediumDisk}s and marks as alternative the items without the best score. * * @param rmds the incoming {@code ReleaseMediumDisk} * @return the processed {@code ReleaseMediumDisk} * ******************************************************************************************************************/ @Nonnull private static List&lt;ReleaseMediumDisk&gt; markedAlternativeByTitleAffinity (@Nonnull final List&lt;ReleaseMediumDisk&gt; rmds) { <span class="fc bfc" id="L688" title="All 2 branches covered."> final int bestScore = rmds.stream().filter(rmd -&gt; !rmd.isAlternative())</span> <span class="fc" id="L689"> .mapToInt(ReleaseMediumDisk::getScore)</span> <span class="fc" id="L690"> .max().getAsInt();</span> <span class="fc bfc" id="L691" title="All 2 branches covered."> return markedAlternative(rmds, rmd -&gt; rmd.getScore() &lt; bestScore);</span> } /******************************************************************************************************************* * * Creates a copy of the collection where items have been marked alternative if the given predicate applies. * * @param rmds the source * @param predicate the predicate to decide whether an item must be marked as alternative * @return the processed collection * ******************************************************************************************************************/ @Nonnull private static List&lt;ReleaseMediumDisk&gt; markedAlternative (@Nonnull final List&lt;ReleaseMediumDisk&gt; rmds, @Nonnull final Predicate&lt;ReleaseMediumDisk&gt; predicate) { <span class="fc" id="L707"> return rmds.stream().map(rmd -&gt; rmd.alternativeIf(predicate.test(rmd))).collect(toList());</span> } /******************************************************************************************************************* * * Finds the first attribute specified by an extractor among items not already marked as alternatives. * * @param rmds the collection to search into * @param extractor the extractor * @return the searched object * ******************************************************************************************************************/ @Nonnull private static &lt;T extends Comparable&lt;?&gt;&gt; Optional&lt;T&gt; findFirstNotInAlternative ( @Nonnull final List&lt;ReleaseMediumDisk&gt; rmds, @Nonnull final Function&lt;ReleaseMediumDisk, Optional&lt;T&gt;&gt; extractor) { <span class="fc" id="L724"> return rmds.stream()</span> <span class="fc bfc" id="L725" title="All 2 branches covered."> .filter(rmd -&gt; !rmd.isAlternative())</span> <span class="fc" id="L726"> .map(extractor)</span> <span class="fc" id="L727"> .flatMap(Optional::stream)</span> <span class="fc" id="L728"> .sorted()</span> <span class="fc" id="L729"> .findFirst();</span> } /******************************************************************************************************************* * ******************************************************************************************************************/ @Nonnegative private static int countOfNotAlternative (@Nonnull final List&lt;ReleaseMediumDisk&gt; rmds) { <span class="fc bfc" id="L738" title="All 2 branches covered."> return (int)rmds.stream().filter(rmd -&gt; !rmd.isAlternative()).count();</span> } /******************************************************************************************************************* * * Extracts data from the given release. For MusicBrainz, a Release is typically a disk, but it can be made of * multiple disks in case of many tracks. * * @param metadata the {@code Metadata} * @param rmd the release * @return the RDF triples * @throws InterruptedException in case of I/O error * @throws IOException in case of I/O error * ******************************************************************************************************************/ @Nonnull private ModelBuilder handleRelease (@Nonnull final Metadata metadata, @Nonnull final ReleaseMediumDisk rmd) throws IOException, InterruptedException { <span class="fc" id="L757"> final Medium medium = rmd.getMedium();</span> <span class="fc" id="L758"> final String releaseId = rmd.getRelease().getId();</span> <span class="fc" id="L759"> final List&lt;DefTrackData&gt; tracks = medium.getTrackList().getDefTrack();</span> <span class="fc" id="L760"> final String embeddedRecordTitle = metadata.get(ALBUM).get(); // .orElse(parent.getPath().toFile().getName());</span> <span class="fc" id="L761"> final Cddb cddb = metadata.get(CDDB).get();</span> <span class="fc" id="L762"> final String recordTitle = rmd.pickTitle();</span> <span class="fc" id="L763"> final IRI embeddedRecordIri = recordIriOf(metadata, embeddedRecordTitle);</span> <span class="fc" id="L764"> final IRI recordIri = BMMO.recordIriFor(rmd.computeId());</span> <span class="fc bfc" id="L765" title="All 2 branches covered."> log.info(&quot;importing {} {} ...&quot;, recordTitle, (rmd.isAlternative() ? &quot;(alternative)&quot; : &quot;&quot;));</span> <span class="fc" id="L767"> ModelBuilder model = createModelBuilder()</span> <span class="fc" id="L768"> .with(recordIri, RDF.TYPE, MO.C_RECORD)</span> <span class="fc" id="L769"> .with(recordIri, RDFS.LABEL, literalFor(recordTitle))</span> <span class="fc" id="L770"> .with(recordIri, DC.TITLE, literalFor(recordTitle))</span> <span class="fc" id="L771"> .with(recordIri, BMMO.P_IMPORTED_FROM, BMMO.O_SOURCE_MUSICBRAINZ)</span> <span class="fc" id="L772"> .with(recordIri, BMMO.P_ALTERNATE_OF, embeddedRecordIri)</span> <span class="fc" id="L773"> .with(recordIri, MO.P_MEDIA_TYPE, MO.C_CD)</span> <span class="fc" id="L774"> .with(recordIri, MO.P_TRACK_COUNT, literalFor(tracks.size()))</span> <span class="fc" id="L775"> .with(recordIri, MO.P_MUSICBRAINZ_GUID, literalFor(releaseId))</span> <span class="fc" id="L776"> .with(recordIri, MO.P_MUSICBRAINZ, musicBrainzIriFor(&quot;release&quot;, releaseId))</span> <span class="fc" id="L777"> .with(recordIri, MO.P_AMAZON_ASIN, literalFor(rmd.getAsin()))</span> <span class="fc" id="L778"> .with(recordIri, MO.P_GTIN, literalFor(rmd.getBarcode()))</span> <span class="fc" id="L780"> .with(tracks.stream().parallel()</span> <span class="fc" id="L781"> .map(_f(track -&gt; handleTrack(rmd, cddb, recordIri, track)))</span> <span class="fc" id="L782"> .collect(toList()));</span> <span class="fc bfc" id="L784" title="All 2 branches covered."> if (rmd.isAlternative())</span> { <span class="fc" id="L786"> model = model.with(recordIri, BMMO.P_ALTERNATE_PICK_OF, embeddedRecordIri);</span> } <span class="fc" id="L789"> return model;</span> // TODO: release.getLabelInfoList(); // TODO: record producer - requires inc=artist-rels } /******************************************************************************************************************* * * Extracts data from the given {@link DefTrackData}. * * @param rmd the release * @param cddb the CDDB of the track we're handling * @param track the track * @return the RDF triples * @throws InterruptedException in case of I/O error * @throws IOException in case of I/O error * ******************************************************************************************************************/ @Nonnull private ModelBuilder handleTrack (@Nonnull final ReleaseMediumDisk rmd, @Nonnull final Cddb cddb, @Nonnull final IRI recordIri, @Nonnull final DefTrackData track) throws IOException, InterruptedException { <span class="fc" id="L813"> final IRI trackIri = trackIriOf(track.getId());</span> <span class="fc" id="L814"> final int trackNumber = track.getPosition().intValue();</span> <span class="fc" id="L815"> final Optional&lt;Integer&gt; diskCount = emptyIfOne(rmd.getDiskCount());</span> <span class="fc" id="L816"> final Optional&lt;Integer&gt; diskNumber = diskCount.flatMap(dc -&gt; rmd.getDiskNumber());</span> <span class="fc" id="L817"> final String recordingId = track.getRecording().getId();</span> // final Recording recording = track.getRecording(); <span class="fc" id="L819"> final Recording recording = mbMetadataProvider.getResource(RECORDING, recordingId, RECORDING_INCLUDES).get();</span> <span class="fc" id="L820"> final String trackTitle = recording.getTitle();</span> // track.getRecording().getAliasList().getAlias().get(0).getSortName(); <span class="fc" id="L822"> final IRI signalIri = signalIriFor(cddb, track.getPosition().intValue());</span> <span class="fc" id="L823"> log.info(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; {}. {}&quot;, trackNumber, trackTitle);</span> <span class="fc" id="L825"> return createModelBuilder()</span> <span class="fc" id="L826"> .with(recordIri, MO.P_TRACK, trackIri)</span> <span class="fc" id="L827"> .with(recordIri, BMMO.P_DISK_COUNT, literalForInt(diskCount))</span> <span class="fc" id="L828"> .with(recordIri, BMMO.P_DISK_NUMBER, literalForInt(diskNumber))</span> <span class="fc" id="L830"> .with(signalIri, MO.P_PUBLISHED_AS, trackIri)</span> <span class="fc" id="L832"> .with(trackIri, RDF.TYPE, MO.C_TRACK)</span> <span class="fc" id="L833"> .with(trackIri, RDFS.LABEL, literalFor(trackTitle))</span> <span class="fc" id="L834"> .with(trackIri, DC.TITLE, literalFor(trackTitle))</span> <span class="fc" id="L835"> .with(trackIri, BMMO.P_IMPORTED_FROM, BMMO.O_SOURCE_MUSICBRAINZ)</span> <span class="fc" id="L836"> .with(trackIri, MO.P_TRACK_NUMBER, literalFor(trackNumber))</span> <span class="fc" id="L837"> .with(trackIri, MO.P_MUSICBRAINZ_GUID, literalFor(track.getId()))</span> <span class="fc" id="L838"> .with(trackIri, MO.P_MUSICBRAINZ, musicBrainzIriFor(&quot;track&quot;, track.getId()))</span> <span class="fc" id="L840"> .with(handleTrackRelations(signalIri, trackIri, recordIri, recording));</span> } /******************************************************************************************************************* * * Extracts data from the relations of the given {@link Recording}. * * @param signalIri the IRI of the signal associated to the track we're handling * @param recording the {@code Recording} * @return the RDF triples * ******************************************************************************************************************/ @Nonnull private ModelBuilder handleTrackRelations (@Nonnull final IRI signalIri, @Nonnull final IRI trackIri, @Nonnull final IRI recordIri, @Nonnull final Recording recording) { <span class="fc" id="L858"> return createModelBuilder().with(recording.getRelationList()</span> <span class="fc" id="L859"> .stream()</span> <span class="fc" id="L860"> .parallel()</span> <span class="fc" id="L861"> .flatMap(RelationAndTargetType::toStream)</span> <span class="fc" id="L862"> .map(ratt -&gt; handleTrackRelation(signalIri, trackIri, recordIri, recording, ratt))</span> <span class="fc" id="L863"> .collect(toList()));</span> } /******************************************************************************************************************* * * Extracts data from a relation of the given {@link Recording}. * * @param signalIri the IRI of the signal associated to the track we're handling * @param recording the {@code Recording} * @param ratt the relation * @return the RDF triples * ******************************************************************************************************************/ @Nonnull private ModelBuilder handleTrackRelation (@Nonnull final IRI signalIri, @Nonnull final IRI trackIri, @Nonnull final IRI recordIri, @Nonnull final Recording recording, @Nonnull final RelationAndTargetType ratt) { <span class="fc" id="L883"> final Relation relation = ratt.getRelation();</span> <span class="fc" id="L884"> final String targetType = ratt.getTargetType();</span> <span class="fc" id="L885"> final List&lt;Attribute&gt; attributes = getAttributes(relation);</span> // final Target target = relation.getTarget(); <span class="fc" id="L887"> final String type = relation.getType();</span> <span class="fc" id="L888"> final Artist artist = relation.getArtist();</span> <span class="fc" id="L890"> log.info(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; {} {} {} {} ({})&quot;, targetType,</span> type, <span class="fc" id="L892"> attributes.stream().map(a -&gt; toString(a)).collect(toList()),</span> <span class="fc" id="L893"> artist.getName(),</span> <span class="fc" id="L894"> artist.getId());</span> <span class="fc" id="L896"> final IRI performanceIri = performanceIriFor(recording.getId());</span> <span class="fc" id="L897"> final IRI artistIri = artistIriOf(artist.getId());</span> <span class="fc" id="L899"> final ModelBuilder model = createModelBuilder()</span> <span class="fc" id="L900"> .with(performanceIri, RDF.TYPE, MO.C_PERFORMANCE)</span> <span class="fc" id="L901"> .with(performanceIri, BMMO.P_IMPORTED_FROM, BMMO.O_SOURCE_MUSICBRAINZ)</span> <span class="fc" id="L902"> .with(performanceIri, MO.P_MUSICBRAINZ_GUID, literalFor(recording.getId()))</span> <span class="fc" id="L903"> .with(performanceIri, MO.P_RECORDED_AS, signalIri)</span> <span class="fc" id="L905"> .with(artistIri, RDF.TYPE, MO.C_MUSIC_ARTIST)</span> <span class="fc" id="L906"> .with(artistIri, RDFS.LABEL, literalFor(artist.getName()))</span> <span class="fc" id="L907"> .with(artistIri, FOAF.NAME, literalFor(artist.getName()))</span> <span class="fc" id="L908"> .with(artistIri, BMMO.P_IMPORTED_FROM, BMMO.O_SOURCE_MUSICBRAINZ)</span> <span class="fc" id="L909"> .with(artistIri, MO.P_MUSICBRAINZ_GUID, literalFor(artist.getId()))</span> <span class="fc" id="L910"> .with(artistIri, MO.P_MUSICBRAINZ, musicBrainzIriFor(&quot;artist&quot;, artist.getId()))</span> // TODO these could be inferred - performance shortcuts. Catalog queries rely upon these. <span class="fc" id="L913"> .with(recordIri, FOAF.MAKER, artistIri)</span> <span class="fc" id="L914"> .with(trackIri, FOAF.MAKER, artistIri)</span> <span class="fc" id="L915"> .with(performanceIri, FOAF.MAKER, artistIri);</span> // .with(signalIri, FOAF.MAKER, artistIri); <span class="pc bpc" id="L918" title="1 of 2 branches missed."> if (&quot;artist&quot;.equals(targetType))</span> { <span class="fc" id="L920"> predicatesForArtists(type, attributes)</span> <span class="fc" id="L921"> .forEach(predicate -&gt; model.with(performanceIri, predicate, artistIri));</span> } <span class="fc" id="L924"> return model;</span> // relation.getBegin(); // relation.getEnd(); // relation.getEnded(); } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static List&lt;IRI&gt; predicatesForArtists (@Nonnull final String type, @Nonnull final List&lt;Attribute&gt; attributes) { <span class="fc bfc" id="L938" title="All 2 branches covered."> if (attributes.isEmpty())</span> { <span class="fc" id="L940"> return singletonList(predicateFor(type));</span> } else { <span class="fc" id="L944"> return attributes.stream().map(attribute -&gt;</span> { <span class="fc" id="L946"> String role = type;</span> <span class="fc bfc" id="L948" title="All 4 branches covered."> if (type.equals(&quot;vocal&quot;) || type.equals(&quot;instrument&quot;))</span> { <span class="fc" id="L950"> role += &quot;/&quot; + attribute.getContent();</span> } <span class="fc" id="L953"> return predicateFor(role);</span> <span class="fc" id="L954"> }).collect(toList());</span> } } /******************************************************************************************************************* * * Given a list of {@link ReleaseGroup}s, navigates into it and extract all CD {@link Medium}s that match the * given CDDB track offsets. * * @param releaseGroups the {@code ReleaseGroup}s * @param cddb the track offsets * @param validation how the results must be validated * @return a collection of filtered {@code Medium}s * ******************************************************************************************************************/ @Nonnull private Collection&lt;ReleaseMediumDisk&gt; findReleases (@Nonnull final List&lt;ReleaseGroup&gt; releaseGroups, @Nonnull final Cddb cddb, @Nonnull final Validation validation) { <span class="fc" id="L974"> return releaseGroups.stream()</span> <span class="fc" id="L975"> .parallel()</span> <span class="fc bfc" id="L976" title="All 2 branches covered."> .filter(releaseGroup -&gt; scoreOf(releaseGroup) &gt;= releaseGroupScoreThreshold)</span> <span class="fc" id="L977"> .peek(this::logArtists)</span> <span class="fc" id="L978"> .map(ReleaseGroup::getReleaseList)</span> <span class="fc" id="L979"> .flatMap(releaseList -&gt; findReleases(releaseList, cddb, validation).stream())</span> <span class="fc" id="L980"> .collect(toList());</span> } /******************************************************************************************************************* * * Given a {@link ReleaseList}, navigates into it and extract all CD {@link Medium}s that match the given CDDB track * offsets. * * @param releaseList the {@code ReleaseList} * @param cddb the track offsets to match * @param validation how the results must be validated * @return a collection of filtered {@code Medium}s * ******************************************************************************************************************/ @Nonnull private Collection&lt;ReleaseMediumDisk&gt; findReleases (@Nonnull final ReleaseList releaseList, @Nonnull final Cddb cddb, @Nonnull final Validation validation) { <span class="fc" id="L999"> return releaseList.getRelease().stream()</span> <span class="fc" id="L1000"> .parallel()</span> // .peek(this::logArtists) <span class="fc" id="L1002"> .peek(release -&gt; log.info(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; release: {} {}&quot;, release.getId(), release.getTitle()))</span> <span class="fc" id="L1003"> .flatMap(_f(release -&gt; mbMetadataProvider.getResource(RELEASE, release.getId(), RELEASE_INCLUDES).get()</span> <span class="fc" id="L1004"> .getMediumList().getMedium()</span> <span class="fc" id="L1005"> .stream()</span> <span class="fc" id="L1006"> .map(medium -&gt; new ReleaseMediumDisk(release, medium))))</span> <span class="fc" id="L1007"> .filter(MusicBrainzAudioMedatataImporter::matchesFormat)</span> <span class="fc" id="L1008"> .flatMap(rmd -&gt; rmd.getMedium().getDiscList().getDisc().stream().map(rmd::withDisc))</span> <span class="fc" id="L1009"> .filter(rmd -&gt; matchesTrackOffsets(rmd, cddb, validation))</span> <span class="fc" id="L1010"> .peek(rmd -&gt; log.info(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; FOUND {} - with score {}&quot;, rmd.getMediumAndDiscString(), 0 /* scoreOf(releaseGroup) FIXME */))</span> <span class="fc" id="L1011"> .collect(toMap(rmd -&gt; rmd.getRelease().getId(), rmd -&gt; rmd, (u, v) -&gt; v, TreeMap::new))</span> <span class="fc" id="L1012"> .values();</span> } /******************************************************************************************************************* * * * * ******************************************************************************************************************/ public static int similarity (@Nonnull final String a, @Nonnull final String b) { <span class="fc" id="L1023"> int score = StringUtils.getFuzzyDistance(a.toLowerCase(), b.toLowerCase(), Locale.UK);</span> // // While this is a hack, it isn't so ugly as it might appear. The idea is to give a lower score to // collections and records with a generic title, hoping that a better one is picked. // FIXME: put into a map and then into an external resource with the delta score associated. // FIXME: with the filtering on collection size, this might be useless? // <span class="fc bfc" id="L1030" title="All 2 branches covered."> if (a.matches(&quot;^Great Violin Concertos.*&quot;)</span> <span class="fc bfc" id="L1031" title="All 2 branches covered."> || a.matches(&quot;^CBS Great Performances.*&quot;))</span> { <span class="fc" id="L1033"> score -= 50;</span> } <span class="fc bfc" id="L1036" title="All 2 branches covered."> if (a.matches(&quot;^Piano Concertos$&quot;)</span> <span class="fc bfc" id="L1037" title="All 2 branches covered."> || a.matches(&quot;^Klavierkonzerte$&quot;))</span> { <span class="fc" id="L1039"> score -= 30;</span> } <span class="fc" id="L1042"> return score;</span> } /******************************************************************************************************************* * * Returns {@code true} if the given {@link ReleaseMediumDisk} is of a meaningful type (that is, a CD) or it's not set. * * @param rmd the {@code ReleaseMediumDisk} * @return {@code true} if there is a match * ******************************************************************************************************************/ private static boolean matchesFormat (@Nonnull final ReleaseMediumDisk rmd) { <span class="fc" id="L1055"> final String format = rmd.getMedium().getFormat();</span> <span class="fc bfc" id="L1057" title="All 4 branches covered."> if ((format != null) &amp;&amp; !&quot;CD&quot;.equals(format))</span> { <span class="fc" id="L1059"> log.info(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; discarded {} because not a CD ({})&quot;, rmd.getMediumAndDiscString(), format);</span> <span class="fc" id="L1060"> return false;</span> } <span class="fc" id="L1063"> return true;</span> } /******************************************************************************************************************* * * Returns {@code true} if the given {@link ReleaseMediumDisk} matches the track offsets in the given {@link Cddb}. * * @param rmd the {@code ReleaseMediumDisk} * @param requestedCddb the track offsets to match * @param validation how the results must be validated * @return {@code true} if there is a match * ******************************************************************************************************************/ private boolean matchesTrackOffsets (@Nonnull final ReleaseMediumDisk rmd, @Nonnull final Cddb requestedCddb, @Nonnull final Validation validation) { <span class="fc" id="L1080"> final Cddb cddb = rmd.getCddb();</span> <span class="pc bpc" id="L1082" title="3 of 4 branches missed."> if ((cddb == null) &amp;&amp; (validation == Validation.TRACK_OFFSETS_MATCH_NOT_REQUIRED))</span> { <span class="nc" id="L1084"> log.info(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; no track offsets, but not required&quot;);</span> <span class="nc" id="L1085"> return true;</span> } <span class="fc" id="L1088"> final boolean matches = requestedCddb.matches(cddb, trackOffsetsMatchThreshold);</span> <span class="fc bfc" id="L1090" title="All 2 branches covered."> if (!matches)</span> { <span class="fc" id="L1092"> synchronized (log) // keep log lines together</span> { <span class="fc" id="L1094"> log.info(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; discarded {} because track offsets don't match&quot;, rmd.getMediumAndDiscString());</span> <span class="fc" id="L1095"> log.debug(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; iTunes offsets: {}&quot;, requestedCddb.getTrackFrameOffsets());</span> <span class="fc" id="L1096"> log.debug(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; found offsets: {}&quot;, cddb.getTrackFrameOffsets());</span> <span class="fc" id="L1097"> }</span> } <span class="fc" id="L1100"> return matches;</span> } /******************************************************************************************************************* * * Searches for an alternate title of a record by querying the embedded title against the CDDB. The CDDB track * offsets are checked to validate the result. * * @param metadata the {@code Metadata} * @return the title, if found * ******************************************************************************************************************/ @Nonnull private Optional&lt;String&gt; cddbAlternateTitleOf (@Nonnull final Metadata metadata) throws IOException, InterruptedException { <span class="fc" id="L1116"> final RestResponse&lt;CddbAlbum&gt; optionalAlbum = cddbMetadataProvider.findCddbAlbum(metadata);</span> <span class="fc bfc" id="L1118" title="All 2 branches covered."> if (!optionalAlbum.isPresent())</span> { <span class="fc" id="L1120"> return Optional.empty();</span> } <span class="fc" id="L1123"> final CddbAlbum album = optionalAlbum.get();</span> <span class="fc" id="L1124"> final Cddb albumCddb = album.getCddb();</span> <span class="fc" id="L1125"> final Cddb requestedCddb = metadata.get(ITUNES_COMMENT).get().getCddb();</span> <span class="fc" id="L1126"> final Optional&lt;String&gt; dTitle = album.getProperty(&quot;DTITLE&quot;);</span> <span class="fc bfc" id="L1128" title="All 2 branches covered."> if (!albumCddb.matches(requestedCddb, trackOffsetsMatchThreshold))</span> { <span class="fc" id="L1130"> synchronized (log) // keep log lines together</span> { <span class="fc" id="L1132"> log.info(&quot;&gt;&gt;&gt;&gt; discarded alternate title because of mismatching track offsets: {}&quot;, dTitle);</span> <span class="fc" id="L1133"> log.debug(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; found track offsets: {}&quot;, albumCddb.getTrackFrameOffsets());</span> <span class="fc" id="L1134"> log.debug(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; searched track offsets: {}&quot;, requestedCddb.getTrackFrameOffsets());</span> <span class="fc" id="L1135"> log.debug(&quot;&gt;&gt;&gt;&gt;&gt;&gt;&gt;&gt; ppm {}&quot;, albumCddb.computeDifference(requestedCddb));</span> <span class="fc" id="L1136"> }</span> <span class="fc" id="L1138"> return Optional.empty();</span> } <span class="fc" id="L1141"> return dTitle;</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static List&lt;Attribute&gt; getAttributes (@Nonnull final Relation relation) { <span class="fc" id="L1152"> final List&lt;Attribute&gt; attributes = new ArrayList&lt;&gt;();</span> <span class="fc bfc" id="L1154" title="All 2 branches covered."> if (relation.getAttributeList() != null)</span> { <span class="fc" id="L1156"> attributes.addAll(relation.getAttributeList().getAttribute());</span> } <span class="fc" id="L1159"> return attributes;</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static ModelBuilder createModelBuilder() { <span class="fc" id="L1170"> return new ModelBuilder(SOURCE_MUSICBRAINZ);</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static IRI artistIriOf (@Nonnull final String id) { <span class="fc" id="L1181"> return BMMO.artistIriFor(createSha1IdNew(musicBrainzIriFor(&quot;artist&quot;, id).stringValue()));</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static IRI trackIriOf (@Nonnull final String id) { <span class="fc" id="L1192"> return BMMO.trackIriFor(createSha1IdNew(musicBrainzIriFor(&quot;track&quot;, id).stringValue()));</span> } /******************************************************************************************************************* * * FIXME: DUPLICATED FROM EmbbededAudioMetadataImporter * ******************************************************************************************************************/ @Nonnull private static IRI recordIriOf (@Nonnull final Metadata metadata, @Nonnull final String recordTitle) { <span class="fc" id="L1203"> final Optional&lt;Cddb&gt; cddb = metadata.get(CDDB);</span> <span class="fc" id="L1204"> return BMMO.recordIriFor(cddb.map(value -&gt; createSha1IdNew(value.getToc()))</span> <span class="pc" id="L1205"> .orElseGet(() -&gt; createSha1IdNew(&quot;RECORD:&quot; + recordTitle)));</span> } /******************************************************************************************************************* * * ******************************************************************************************************************/ @Nonnull private IRI signalIriFor (@Nonnull final Cddb cddb, @Nonnegative final int trackNumber) { <span class="fc" id="L1215"> return BMMO.signalIriFor(createSha1IdNew(cddb.getToc() + &quot;/&quot; + trackNumber));</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static IRI performanceIriFor (@Nonnull final String id) { <span class="fc" id="L1226"> return BMMO.performanceIriFor(createSha1IdNew(musicBrainzIriFor(&quot;performance&quot;, id).stringValue()));</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static IRI musicBrainzIriFor (@Nonnull final String resourceType, @Nonnull final String id) { <span class="fc" id="L1237"> return FACTORY.createIRI(String.format(&quot;http://musicbrainz.org/%s/%s&quot;, resourceType, id));</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static IRI predicateFor (@Nonnull final String role) { <span class="fc" id="L1248"> return Objects.requireNonNull(PERFORMER_MAP.get(role.toLowerCase()), &quot;Cannot map role: &quot; + role);</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ private static int scoreOf (@Nonnull final ReleaseGroup releaseGroup) { <span class="fc" id="L1258"> return Integer.parseInt(releaseGroup.getOtherAttributes().get(QNAME_SCORE));</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ private void logArtists (@Nonnull final ReleaseGroup releaseGroup) { <span class="fc" id="L1268"> log.debug(&quot;&gt;&gt;&gt;&gt; {} {} {} artist: {}&quot;,</span> <span class="fc" id="L1269"> releaseGroup.getOtherAttributes().get(QNAME_SCORE),</span> <span class="fc" id="L1270"> releaseGroup.getId(),</span> <span class="fc" id="L1271"> releaseGroup.getTitle(),</span> <span class="fc" id="L1272"> releaseGroup.getArtistCredit().getNameCredit().stream().map(nc -&gt; nc.getArtist().getName()).collect(toList()));</span> <span class="fc" id="L1273"> }</span> /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static Optional&lt;Integer&gt; emptyIfOne (@Nonnull final Optional&lt;Integer&gt; number) { <span class="fc bfc" id="L1283" title="All 2 branches covered."> return number.flatMap(n -&gt; (n == 1) ? Optional.empty() : Optional.of(n));</span> } /******************************************************************************************************************* * * * ******************************************************************************************************************/ @Nonnull private static String toString (@Nonnull final Attribute attribute) { <span class="fc" id="L1294"> return String.format(&quot;%s %s (%s)&quot;, attribute.getContent(), attribute.getCreditedAs(), attribute.getValue());</span> } } </pre><div class="footer"><span class="right">Created with <a href="http://www.jacoco.org/jacoco">JaCoCo</a> 0.8.7.202105040129</span></div></body></html>