/* * (C) 2007-2012 Alibaba Group Holding Limited. * * Licensed under the Apache License, Version 2.0 (the "License"); * 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 "AS IS" 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. * Authors: * wuhua <wq163@163.com> , boyan <killme2008@gmail.com> */ package com.taobao.metamorphosis.client.consumer.storage; import java.io.BufferedReader; import java.io.Closeable; import java.io.File; import java.io.FileOutputStream; import java.io.FileReader; import java.io.IOException; import java.nio.ByteBuffer; import java.nio.channels.FileChannel; import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.atomic.AtomicLong; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import com.taobao.metamorphosis.client.consumer.TopicPartitionRegInfo; import com.taobao.metamorphosis.cluster.Partition; import com.taobao.metamorphosis.utils.JSONUtils; /** * ����offset�洢���洢�ڴ��̣�Ĭ�ϴ洢��$HOME/.meta_offsets�ļ��� * * @author boyan * @Date 2011-5-4 * */ public class LocalOffsetStorage implements OffsetStorage { static final Log log = LogFactory.getLog(LocalOffsetStorage.class); private final Map<String/* group */, List<TopicPartitionRegInfo>> groupInfoMap = new HashMap<String, List<TopicPartitionRegInfo>>(); private final AtomicLong counter = new AtomicLong(); private final String filePath; public LocalOffsetStorage() throws IOException { this(System.getProperty("user.home") + File.separator + ".meta_offsets"); } public LocalOffsetStorage(final String filePath) throws IOException { this.filePath = filePath; final File file = new File(filePath); if (file.exists()) { this.loadGroupInfo(file); } else { file.createNewFile(); } } private void loadGroupInfo(final File file) { String line = null; BufferedReader reader = null; FileReader fileReader = null; final StringBuilder jsonSB = new StringBuilder(); try { fileReader = new FileReader(file); reader = new BufferedReader(fileReader); while ((line = reader.readLine()) != null) { jsonSB.append(line); } } catch (final IOException e) { log.error("��ȡ�ļ�" + file + "����", e); } finally { this.close(reader); this.close(fileReader); } try { // ��ֹ�ļ�����Ϊ��ʱ������json�����л��쳣,add by wuhua if (jsonSB.length() <= 0) { log.warn(file.getAbsolutePath() + "�ļ�����Ϊ��,��ʱδ���ص�offset��Ϣ,����ǵ�һ�η���������������"); return; } final Map<String/* group */, List<Map<String, Object>>> groupInfoStringMap = (Map<String/* group */, List<Map<String, Object>>>) JSONUtils.deserializeObject(jsonSB.toString(), ConcurrentHashMap.class); for (final Map.Entry<String, List<Map<String, Object>>> entry1 : groupInfoStringMap.entrySet()) { final String group = entry1.getKey(); final List<Map<String, Object>> infos = entry1.getValue(); final List<TopicPartitionRegInfo> infoList = new ArrayList<TopicPartitionRegInfo>(); if (infos != null) { for (final Map<String, Object> infoMap : infos) { final String topic = (String) infoMap.get("topic"); final long offset = Long.valueOf(String.valueOf(infoMap.get("offset"))); final Map<String, Integer> partMap = (Map<String, Integer>) infoMap.get("partition"); infoList.add(new TopicPartitionRegInfo(topic, new Partition(partMap.get("brokerId"), partMap .get("partition")), offset)); } } this.groupInfoMap.put(group, infoList); } } catch (final Exception e) { log.error("�����л�jsonʧ��", e); } } private void close(final Closeable closeable) { try { if (closeable != null) { closeable.close(); } } catch (final IOException e) { // ignore } } @Override public void close() { // do nothing. } @Override public void commitOffset(final String group, final Collection<TopicPartitionRegInfo> infoList) { if (infoList == null || infoList.isEmpty()) { return; } this.groupInfoMap.put(group, (List<TopicPartitionRegInfo>) infoList); FileOutputStream out = null; FileChannel channel = null; try { final String json = JSONUtils.serializeObject(this.groupInfoMap); // write to temp file File tmpFile = new File(this.filePath + ".tmp." + this.counter.incrementAndGet()); out = new FileOutputStream(tmpFile); channel = out.getChannel(); final ByteBuffer buf = ByteBuffer.wrap(json.getBytes()); while (buf.hasRemaining()) { channel.write(buf); } this.close(channel); this.close(out); // rename temp file to target file. synchronized (this) { if (!tmpFile.renameTo(new File(this.filePath))) { throw new IOException("Could not rename temp file to " + this.filePath); } } } catch (final Exception e) { log.error("commitOffset failed ", e); } finally { if (channel != null && channel.isOpen()) { this.close(channel); } this.close(out); } } @Override public void initOffset(final String topic, final String group, final Partition partition, final long offset) { // do nothing } @Override public TopicPartitionRegInfo load(final String topic, final String group, final Partition partition) { final Collection<TopicPartitionRegInfo> topicPartitionRegInfos = this.groupInfoMap.get(group); if (topicPartitionRegInfos == null || topicPartitionRegInfos.isEmpty()) { return null; } for (final TopicPartitionRegInfo info : topicPartitionRegInfos) { if (info.getTopic().equals(topic) && info.getPartition().equals(partition)) { return info; } } return null; } }